2019-05-27 09:55:01 +03:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2005-04-17 02:20:36 +04:00
|
|
|
/*
|
|
|
|
* Scatterlist Cryptographic API.
|
|
|
|
*
|
|
|
|
* Copyright (c) 2002 James Morris <jmorris@intercode.com.au>
|
|
|
|
* Copyright (c) 2002 David S. Miller (davem@redhat.com)
|
2005-11-05 08:58:14 +03:00
|
|
|
* Copyright (c) 2005 Herbert Xu <herbert@gondor.apana.org.au>
|
2005-04-17 02:20:36 +04:00
|
|
|
*
|
|
|
|
* Portions derived from Cryptoapi, by Alexander Kjeldaas <astor@fast.no>
|
2007-10-20 01:06:17 +04:00
|
|
|
* and Nettle, by Niels Möller.
|
2005-04-17 02:20:36 +04:00
|
|
|
*/
|
2005-07-07 00:54:31 +04:00
|
|
|
|
2006-09-21 05:39:29 +04:00
|
|
|
#include <linux/err.h>
|
2005-04-17 02:20:36 +04:00
|
|
|
#include <linux/errno.h>
|
2005-11-05 08:58:14 +03:00
|
|
|
#include <linux/kernel.h>
|
2005-07-07 00:53:09 +04:00
|
|
|
#include <linux/kmod.h>
|
2006-09-21 05:31:44 +04:00
|
|
|
#include <linux/module.h>
|
2006-08-06 15:23:26 +04:00
|
|
|
#include <linux/param.h>
|
2017-02-02 21:15:33 +03:00
|
|
|
#include <linux/sched/signal.h>
|
2005-04-17 02:20:36 +04:00
|
|
|
#include <linux/slab.h>
|
2005-11-05 08:58:14 +03:00
|
|
|
#include <linux/string.h>
|
2017-10-18 10:00:38 +03:00
|
|
|
#include <linux/completion.h>
|
2005-04-17 02:20:36 +04:00
|
|
|
#include "internal.h"
|
|
|
|
|
|
|
|
LIST_HEAD(crypto_alg_list);
|
2006-08-21 15:08:13 +04:00
|
|
|
EXPORT_SYMBOL_GPL(crypto_alg_list);
|
2005-04-17 02:20:36 +04:00
|
|
|
DECLARE_RWSEM(crypto_alg_sem);
|
2006-08-21 15:08:13 +04:00
|
|
|
EXPORT_SYMBOL_GPL(crypto_alg_sem);
|
2005-04-17 02:20:36 +04:00
|
|
|
|
2006-08-06 15:23:26 +04:00
|
|
|
BLOCKING_NOTIFIER_HEAD(crypto_chain);
|
|
|
|
EXPORT_SYMBOL_GPL(crypto_chain);
|
|
|
|
|
2013-09-08 08:33:50 +04:00
|
|
|
static struct crypto_alg *crypto_larval_wait(struct crypto_alg *alg);
|
|
|
|
|
2006-08-06 15:23:26 +04:00
|
|
|
struct crypto_alg *crypto_mod_get(struct crypto_alg *alg)
|
2006-08-06 14:28:44 +04:00
|
|
|
{
|
|
|
|
return try_module_get(alg->cra_module) ? crypto_alg_get(alg) : NULL;
|
2005-04-17 02:20:36 +04:00
|
|
|
}
|
2006-08-06 15:23:26 +04:00
|
|
|
EXPORT_SYMBOL_GPL(crypto_mod_get);
|
2005-04-17 02:20:36 +04:00
|
|
|
|
2006-08-06 15:23:26 +04:00
|
|
|
void crypto_mod_put(struct crypto_alg *alg)
|
2005-04-17 02:20:36 +04:00
|
|
|
{
|
2007-05-19 08:51:00 +04:00
|
|
|
struct module *module = alg->cra_module;
|
|
|
|
|
2006-08-06 14:28:44 +04:00
|
|
|
crypto_alg_put(alg);
|
2007-05-19 08:51:00 +04:00
|
|
|
module_put(module);
|
2005-04-17 02:20:36 +04:00
|
|
|
}
|
2006-08-06 15:23:26 +04:00
|
|
|
EXPORT_SYMBOL_GPL(crypto_mod_put);
|
2005-04-17 02:20:36 +04:00
|
|
|
|
2008-08-03 17:15:23 +04:00
|
|
|
static inline int crypto_is_test_larval(struct crypto_larval *larval)
|
|
|
|
{
|
|
|
|
return larval->alg.cra_driver_name[0];
|
|
|
|
}
|
|
|
|
|
2008-08-04 07:44:59 +04:00
|
|
|
static struct crypto_alg *__crypto_alg_lookup(const char *name, u32 type,
|
|
|
|
u32 mask)
|
2005-04-17 02:20:36 +04:00
|
|
|
{
|
|
|
|
struct crypto_alg *q, *alg = NULL;
|
2006-08-06 15:23:26 +04:00
|
|
|
int best = -2;
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
list_for_each_entry(q, &crypto_alg_list, cra_list) {
|
2005-11-05 08:58:14 +03:00
|
|
|
int exact, fuzzy;
|
|
|
|
|
2006-09-21 05:39:29 +04:00
|
|
|
if (crypto_is_moribund(q))
|
|
|
|
continue;
|
|
|
|
|
2006-09-21 05:35:17 +04:00
|
|
|
if ((q->cra_flags ^ type) & mask)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (crypto_is_larval(q) &&
|
2008-08-03 17:15:23 +04:00
|
|
|
!crypto_is_test_larval((struct crypto_larval *)q) &&
|
2006-09-21 05:35:17 +04:00
|
|
|
((struct crypto_larval *)q)->mask != mask)
|
|
|
|
continue;
|
|
|
|
|
2005-11-05 08:58:14 +03:00
|
|
|
exact = !strcmp(q->cra_driver_name, name);
|
|
|
|
fuzzy = !strcmp(q->cra_name, name);
|
|
|
|
if (!exact && !(fuzzy && q->cra_priority > best))
|
|
|
|
continue;
|
|
|
|
|
2006-05-28 03:05:24 +04:00
|
|
|
if (unlikely(!crypto_mod_get(q)))
|
2005-11-05 08:58:14 +03:00
|
|
|
continue;
|
|
|
|
|
|
|
|
best = q->cra_priority;
|
|
|
|
if (alg)
|
2006-05-28 03:05:24 +04:00
|
|
|
crypto_mod_put(alg);
|
2005-11-05 08:58:14 +03:00
|
|
|
alg = q;
|
|
|
|
|
|
|
|
if (exact)
|
2005-04-17 02:20:36 +04:00
|
|
|
break;
|
|
|
|
}
|
2006-08-06 15:23:26 +04:00
|
|
|
|
|
|
|
return alg;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void crypto_larval_destroy(struct crypto_alg *alg)
|
|
|
|
{
|
|
|
|
struct crypto_larval *larval = (void *)alg;
|
|
|
|
|
|
|
|
BUG_ON(!crypto_is_larval(alg));
|
2019-12-11 05:50:11 +03:00
|
|
|
if (!IS_ERR_OR_NULL(larval->adult))
|
2006-08-06 15:23:26 +04:00
|
|
|
crypto_mod_put(larval->adult);
|
|
|
|
kfree(larval);
|
|
|
|
}
|
|
|
|
|
2008-08-03 17:15:23 +04:00
|
|
|
struct crypto_larval *crypto_larval_alloc(const char *name, u32 type, u32 mask)
|
2006-08-06 15:23:26 +04:00
|
|
|
{
|
|
|
|
struct crypto_larval *larval;
|
|
|
|
|
|
|
|
larval = kzalloc(sizeof(*larval), GFP_KERNEL);
|
|
|
|
if (!larval)
|
2006-09-21 05:39:29 +04:00
|
|
|
return ERR_PTR(-ENOMEM);
|
2006-08-06 15:23:26 +04:00
|
|
|
|
2006-09-21 05:35:17 +04:00
|
|
|
larval->mask = mask;
|
|
|
|
larval->alg.cra_flags = CRYPTO_ALG_LARVAL | type;
|
2006-08-06 15:23:26 +04:00
|
|
|
larval->alg.cra_priority = -1;
|
|
|
|
larval->alg.cra_destroy = crypto_larval_destroy;
|
|
|
|
|
|
|
|
strlcpy(larval->alg.cra_name, name, CRYPTO_MAX_ALG_NAME);
|
|
|
|
init_completion(&larval->completion);
|
|
|
|
|
2008-08-03 17:15:23 +04:00
|
|
|
return larval;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(crypto_larval_alloc);
|
|
|
|
|
|
|
|
static struct crypto_alg *crypto_larval_add(const char *name, u32 type,
|
|
|
|
u32 mask)
|
|
|
|
{
|
|
|
|
struct crypto_alg *alg;
|
|
|
|
struct crypto_larval *larval;
|
|
|
|
|
|
|
|
larval = crypto_larval_alloc(name, type, mask);
|
|
|
|
if (IS_ERR(larval))
|
|
|
|
return ERR_CAST(larval);
|
|
|
|
|
2017-12-29 19:00:46 +03:00
|
|
|
refcount_set(&larval->alg.cra_refcnt, 2);
|
2008-08-03 17:15:23 +04:00
|
|
|
|
2006-08-06 15:23:26 +04:00
|
|
|
down_write(&crypto_alg_sem);
|
2006-09-21 05:35:17 +04:00
|
|
|
alg = __crypto_alg_lookup(name, type, mask);
|
2006-08-06 15:23:26 +04:00
|
|
|
if (!alg) {
|
|
|
|
alg = &larval->alg;
|
|
|
|
list_add(&alg->cra_list, &crypto_alg_list);
|
|
|
|
}
|
|
|
|
up_write(&crypto_alg_sem);
|
|
|
|
|
2013-09-08 08:33:50 +04:00
|
|
|
if (alg != &larval->alg) {
|
2006-08-06 15:23:26 +04:00
|
|
|
kfree(larval);
|
2013-09-08 08:33:50 +04:00
|
|
|
if (crypto_is_larval(alg))
|
|
|
|
alg = crypto_larval_wait(alg);
|
|
|
|
}
|
2006-08-06 15:23:26 +04:00
|
|
|
|
|
|
|
return alg;
|
|
|
|
}
|
|
|
|
|
2007-12-04 04:46:48 +03:00
|
|
|
void crypto_larval_kill(struct crypto_alg *alg)
|
2006-08-06 15:23:26 +04:00
|
|
|
{
|
|
|
|
struct crypto_larval *larval = (void *)alg;
|
|
|
|
|
|
|
|
down_write(&crypto_alg_sem);
|
|
|
|
list_del(&alg->cra_list);
|
|
|
|
up_write(&crypto_alg_sem);
|
2007-05-19 11:51:40 +04:00
|
|
|
complete_all(&larval->completion);
|
2006-08-06 15:23:26 +04:00
|
|
|
crypto_alg_put(alg);
|
|
|
|
}
|
2007-12-04 04:46:48 +03:00
|
|
|
EXPORT_SYMBOL_GPL(crypto_larval_kill);
|
2006-08-06 15:23:26 +04:00
|
|
|
|
|
|
|
static struct crypto_alg *crypto_larval_wait(struct crypto_alg *alg)
|
|
|
|
{
|
|
|
|
struct crypto_larval *larval = (void *)alg;
|
2008-08-03 17:15:23 +04:00
|
|
|
long timeout;
|
|
|
|
|
2015-10-19 13:23:57 +03:00
|
|
|
timeout = wait_for_completion_killable_timeout(
|
2008-08-03 17:15:23 +04:00
|
|
|
&larval->completion, 60 * HZ);
|
2006-08-06 15:23:26 +04:00
|
|
|
|
|
|
|
alg = larval->adult;
|
2008-08-03 17:15:23 +04:00
|
|
|
if (timeout < 0)
|
|
|
|
alg = ERR_PTR(-EINTR);
|
|
|
|
else if (!timeout)
|
|
|
|
alg = ERR_PTR(-ETIMEDOUT);
|
|
|
|
else if (!alg)
|
2006-09-21 05:39:29 +04:00
|
|
|
alg = ERR_PTR(-ENOENT);
|
2019-12-11 05:50:11 +03:00
|
|
|
else if (IS_ERR(alg))
|
|
|
|
;
|
2008-08-03 17:15:23 +04:00
|
|
|
else if (crypto_is_test_larval(larval) &&
|
|
|
|
!(alg->cra_flags & CRYPTO_ALG_TESTED))
|
|
|
|
alg = ERR_PTR(-EAGAIN);
|
|
|
|
else if (!crypto_mod_get(alg))
|
|
|
|
alg = ERR_PTR(-EAGAIN);
|
2006-08-06 15:23:26 +04:00
|
|
|
crypto_mod_put(&larval->alg);
|
|
|
|
|
|
|
|
return alg;
|
|
|
|
}
|
|
|
|
|
2018-03-20 03:05:39 +03:00
|
|
|
static struct crypto_alg *crypto_alg_lookup(const char *name, u32 type,
|
|
|
|
u32 mask)
|
2006-08-06 15:23:26 +04:00
|
|
|
{
|
|
|
|
struct crypto_alg *alg;
|
2018-03-20 10:52:45 +03:00
|
|
|
u32 test = 0;
|
|
|
|
|
|
|
|
if (!((type | mask) & CRYPTO_ALG_TESTED))
|
|
|
|
test |= CRYPTO_ALG_TESTED;
|
2006-08-06 15:23:26 +04:00
|
|
|
|
|
|
|
down_read(&crypto_alg_sem);
|
2018-03-20 10:52:45 +03:00
|
|
|
alg = __crypto_alg_lookup(name, type | test, mask | test);
|
2018-04-17 02:59:13 +03:00
|
|
|
if (!alg && test) {
|
|
|
|
alg = __crypto_alg_lookup(name, type, mask);
|
|
|
|
if (alg && !crypto_is_larval(alg)) {
|
|
|
|
/* Test failed */
|
|
|
|
crypto_mod_put(alg);
|
|
|
|
alg = ERR_PTR(-ELIBBAD);
|
|
|
|
}
|
|
|
|
}
|
2005-04-17 02:20:36 +04:00
|
|
|
up_read(&crypto_alg_sem);
|
2006-08-06 15:23:26 +04:00
|
|
|
|
2005-04-17 02:20:36 +04:00
|
|
|
return alg;
|
|
|
|
}
|
|
|
|
|
2017-12-07 21:55:59 +03:00
|
|
|
static struct crypto_alg *crypto_larval_lookup(const char *name, u32 type,
|
|
|
|
u32 mask)
|
2005-07-07 00:53:09 +04:00
|
|
|
{
|
2006-08-06 15:23:26 +04:00
|
|
|
struct crypto_alg *alg;
|
|
|
|
|
2006-09-21 05:39:29 +04:00
|
|
|
if (!name)
|
|
|
|
return ERR_PTR(-ENOENT);
|
|
|
|
|
2016-11-22 15:08:21 +03:00
|
|
|
type &= ~(CRYPTO_ALG_LARVAL | CRYPTO_ALG_DEAD);
|
2006-09-21 05:39:29 +04:00
|
|
|
mask &= ~(CRYPTO_ALG_LARVAL | CRYPTO_ALG_DEAD);
|
2006-09-21 05:35:17 +04:00
|
|
|
|
2009-02-26 09:06:31 +03:00
|
|
|
alg = crypto_alg_lookup(name, type, mask);
|
2018-06-09 00:57:42 +03:00
|
|
|
if (!alg && !(mask & CRYPTO_NOLOAD)) {
|
2014-11-21 04:05:53 +03:00
|
|
|
request_module("crypto-%s", name);
|
2009-02-26 09:06:31 +03:00
|
|
|
|
2009-04-21 09:27:16 +04:00
|
|
|
if (!((type ^ CRYPTO_ALG_NEED_FALLBACK) & mask &
|
2009-06-02 08:13:14 +04:00
|
|
|
CRYPTO_ALG_NEED_FALLBACK))
|
2014-11-21 04:05:53 +03:00
|
|
|
request_module("crypto-%s-all", name);
|
2009-02-26 09:06:31 +03:00
|
|
|
|
|
|
|
alg = crypto_alg_lookup(name, type, mask);
|
|
|
|
}
|
|
|
|
|
2018-03-20 10:52:45 +03:00
|
|
|
if (!IS_ERR_OR_NULL(alg) && crypto_is_larval(alg))
|
|
|
|
alg = crypto_larval_wait(alg);
|
|
|
|
else if (!alg)
|
|
|
|
alg = crypto_larval_add(name, type, mask);
|
2006-08-06 15:23:26 +04:00
|
|
|
|
2018-03-20 10:52:45 +03:00
|
|
|
return alg;
|
2007-12-04 04:46:48 +03:00
|
|
|
}
|
|
|
|
|
2008-08-03 17:15:23 +04:00
|
|
|
int crypto_probing_notify(unsigned long val, void *v)
|
|
|
|
{
|
|
|
|
int ok;
|
|
|
|
|
|
|
|
ok = blocking_notifier_call_chain(&crypto_chain, val, v);
|
|
|
|
if (ok == NOTIFY_DONE) {
|
|
|
|
request_module("cryptomgr");
|
|
|
|
ok = blocking_notifier_call_chain(&crypto_chain, val, v);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ok;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(crypto_probing_notify);
|
|
|
|
|
2007-12-04 04:46:48 +03:00
|
|
|
struct crypto_alg *crypto_alg_mod_lookup(const char *name, u32 type, u32 mask)
|
|
|
|
{
|
|
|
|
struct crypto_alg *alg;
|
|
|
|
struct crypto_alg *larval;
|
|
|
|
int ok;
|
|
|
|
|
2015-03-30 22:55:52 +03:00
|
|
|
/*
|
|
|
|
* If the internal flag is set for a cipher, require a caller to
|
|
|
|
* to invoke the cipher with the internal flag to use that cipher.
|
|
|
|
* Also, if a caller wants to allocate a cipher that may or may
|
|
|
|
* not be an internal cipher, use type | CRYPTO_ALG_INTERNAL and
|
|
|
|
* !(mask & CRYPTO_ALG_INTERNAL).
|
|
|
|
*/
|
|
|
|
if (!((type | mask) & CRYPTO_ALG_INTERNAL))
|
|
|
|
mask |= CRYPTO_ALG_INTERNAL;
|
|
|
|
|
2007-12-04 04:46:48 +03:00
|
|
|
larval = crypto_larval_lookup(name, type, mask);
|
2006-09-21 05:39:29 +04:00
|
|
|
if (IS_ERR(larval) || !crypto_is_larval(larval))
|
2006-08-06 15:23:26 +04:00
|
|
|
return larval;
|
|
|
|
|
2008-08-03 17:15:23 +04:00
|
|
|
ok = crypto_probing_notify(CRYPTO_MSG_ALG_REQUEST, larval);
|
2006-09-21 05:31:44 +04:00
|
|
|
|
|
|
|
if (ok == NOTIFY_STOP)
|
2006-08-06 15:23:26 +04:00
|
|
|
alg = crypto_larval_wait(larval);
|
|
|
|
else {
|
|
|
|
crypto_mod_put(larval);
|
2006-09-21 05:39:29 +04:00
|
|
|
alg = ERR_PTR(-ENOENT);
|
2006-08-06 15:23:26 +04:00
|
|
|
}
|
|
|
|
crypto_larval_kill(larval);
|
|
|
|
return alg;
|
2005-07-07 00:53:09 +04:00
|
|
|
}
|
2006-09-21 05:35:17 +04:00
|
|
|
EXPORT_SYMBOL_GPL(crypto_alg_mod_lookup);
|
2005-07-07 00:53:09 +04:00
|
|
|
|
2007-01-24 12:50:26 +03:00
|
|
|
static int crypto_init_ops(struct crypto_tfm *tfm, u32 type, u32 mask)
|
2005-04-17 02:20:36 +04:00
|
|
|
{
|
2007-01-24 12:50:26 +03:00
|
|
|
const struct crypto_type *type_obj = tfm->__crt_alg->cra_type;
|
2006-08-21 18:06:54 +04:00
|
|
|
|
2007-01-24 12:50:26 +03:00
|
|
|
if (type_obj)
|
|
|
|
return type_obj->init(tfm, type, mask);
|
crypto: cipher - remove crt_u.cipher (struct cipher_tfm)
Of the three fields in crt_u.cipher (struct cipher_tfm), ->cit_setkey()
is pointless because it always points to setkey() in crypto/cipher.c.
->cit_decrypt_one() and ->cit_encrypt_one() are slightly less pointless,
since if the algorithm doesn't have an alignmask, they are set directly
to ->cia_encrypt() and ->cia_decrypt(). However, this "optimization"
isn't worthwhile because:
- The "cipher" algorithm type is the only algorithm still using crt_u,
so it's bloating every struct crypto_tfm for every algorithm type.
- If the algorithm has an alignmask, this "optimization" actually makes
things slower, as it causes 2 indirect calls per block rather than 1.
- It adds extra code complexity.
- Some templates already call ->cia_encrypt()/->cia_decrypt() directly
instead of going through ->cit_encrypt_one()/->cit_decrypt_one().
- The "cipher" algorithm type never gives optimal performance anyway.
For that, a higher-level type such as skcipher needs to be used.
Therefore, just remove the extra indirection, and make
crypto_cipher_setkey(), crypto_cipher_encrypt_one(), and
crypto_cipher_decrypt_one() be direct calls into crypto/cipher.c.
Also remove the unused function crypto_cipher_cast().
Signed-off-by: Eric Biggers <ebiggers@google.com>
Signed-off-by: Herbert Xu <herbert@gondor.apana.org.au>
2019-12-03 00:42:30 +03:00
|
|
|
return 0;
|
2005-04-17 02:20:36 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
static void crypto_exit_ops(struct crypto_tfm *tfm)
|
|
|
|
{
|
2006-08-21 18:06:54 +04:00
|
|
|
const struct crypto_type *type = tfm->__crt_alg->cra_type;
|
|
|
|
|
2016-10-08 00:13:35 +03:00
|
|
|
if (type && tfm->exit)
|
|
|
|
tfm->exit(tfm);
|
2005-04-17 02:20:36 +04:00
|
|
|
}
|
|
|
|
|
2007-01-24 12:50:26 +03:00
|
|
|
static unsigned int crypto_ctxsize(struct crypto_alg *alg, u32 type, u32 mask)
|
2005-07-07 00:53:29 +04:00
|
|
|
{
|
2007-01-24 12:50:26 +03:00
|
|
|
const struct crypto_type *type_obj = alg->cra_type;
|
2005-07-07 00:53:29 +04:00
|
|
|
unsigned int len;
|
|
|
|
|
2006-08-21 18:06:54 +04:00
|
|
|
len = alg->cra_alignmask & ~(crypto_tfm_ctx_alignment() - 1);
|
2007-01-24 12:50:26 +03:00
|
|
|
if (type_obj)
|
|
|
|
return len + type_obj->ctxsize(alg, type, mask);
|
2006-08-21 18:06:54 +04:00
|
|
|
|
2005-07-07 00:53:29 +04:00
|
|
|
switch (alg->cra_flags & CRYPTO_ALG_TYPE_MASK) {
|
|
|
|
default:
|
|
|
|
BUG();
|
|
|
|
|
|
|
|
case CRYPTO_ALG_TYPE_CIPHER:
|
2007-01-27 02:05:15 +03:00
|
|
|
len += crypto_cipher_ctxsize(alg);
|
2005-07-07 00:53:29 +04:00
|
|
|
break;
|
2009-07-12 09:58:04 +04:00
|
|
|
|
2005-07-07 00:53:29 +04:00
|
|
|
case CRYPTO_ALG_TYPE_COMPRESS:
|
2007-01-27 02:05:15 +03:00
|
|
|
len += crypto_compress_ctxsize(alg);
|
2005-07-07 00:53:29 +04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2006-08-21 18:06:54 +04:00
|
|
|
return len;
|
2005-07-07 00:53:29 +04:00
|
|
|
}
|
|
|
|
|
2019-12-07 17:15:15 +03:00
|
|
|
static void crypto_shoot_alg(struct crypto_alg *alg)
|
2006-09-21 05:39:29 +04:00
|
|
|
{
|
|
|
|
down_write(&crypto_alg_sem);
|
|
|
|
alg->cra_flags |= CRYPTO_ALG_DYING;
|
|
|
|
up_write(&crypto_alg_sem);
|
|
|
|
}
|
|
|
|
|
2007-01-24 12:50:26 +03:00
|
|
|
struct crypto_tfm *__crypto_alloc_tfm(struct crypto_alg *alg, u32 type,
|
|
|
|
u32 mask)
|
2005-04-17 02:20:36 +04:00
|
|
|
{
|
|
|
|
struct crypto_tfm *tfm = NULL;
|
2005-07-07 00:53:29 +04:00
|
|
|
unsigned int tfm_size;
|
2006-09-21 05:39:29 +04:00
|
|
|
int err = -ENOMEM;
|
2005-07-07 00:53:29 +04:00
|
|
|
|
2007-01-24 12:50:26 +03:00
|
|
|
tfm_size = sizeof(*tfm) + crypto_ctxsize(alg, type, mask);
|
2006-03-06 13:42:07 +03:00
|
|
|
tfm = kzalloc(tfm_size, GFP_KERNEL);
|
2005-04-17 02:20:36 +04:00
|
|
|
if (tfm == NULL)
|
2006-10-11 16:29:51 +04:00
|
|
|
goto out_err;
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
tfm->__crt_alg = alg;
|
2006-09-21 05:39:29 +04:00
|
|
|
|
2007-01-24 12:50:26 +03:00
|
|
|
err = crypto_init_ops(tfm, type, mask);
|
2006-09-21 05:39:29 +04:00
|
|
|
if (err)
|
2005-04-17 02:20:36 +04:00
|
|
|
goto out_free_tfm;
|
2006-05-24 07:02:26 +04:00
|
|
|
|
2008-09-14 05:19:03 +04:00
|
|
|
if (!tfm->exit && alg->cra_init && (err = alg->cra_init(tfm)))
|
2006-05-24 07:02:26 +04:00
|
|
|
goto cra_init_failed;
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
goto out;
|
|
|
|
|
2006-05-24 07:02:26 +04:00
|
|
|
cra_init_failed:
|
|
|
|
crypto_exit_ops(tfm);
|
2005-04-17 02:20:36 +04:00
|
|
|
out_free_tfm:
|
2008-09-14 05:19:03 +04:00
|
|
|
if (err == -EAGAIN)
|
|
|
|
crypto_shoot_alg(alg);
|
2005-04-17 02:20:36 +04:00
|
|
|
kfree(tfm);
|
2006-10-11 16:29:51 +04:00
|
|
|
out_err:
|
2006-09-21 05:39:29 +04:00
|
|
|
tfm = ERR_PTR(err);
|
2005-04-17 02:20:36 +04:00
|
|
|
out:
|
|
|
|
return tfm;
|
|
|
|
}
|
2006-09-21 05:39:29 +04:00
|
|
|
EXPORT_SYMBOL_GPL(__crypto_alloc_tfm);
|
|
|
|
|
2006-07-30 05:53:01 +04:00
|
|
|
/*
|
|
|
|
* crypto_alloc_base - Locate algorithm and allocate transform
|
|
|
|
* @alg_name: Name of algorithm
|
|
|
|
* @type: Type of algorithm
|
|
|
|
* @mask: Mask for type comparison
|
|
|
|
*
|
2008-09-21 01:52:53 +04:00
|
|
|
* This function should not be used by new algorithm types.
|
2013-06-28 16:56:20 +04:00
|
|
|
* Please use crypto_alloc_tfm instead.
|
2008-09-21 01:52:53 +04:00
|
|
|
*
|
2006-07-30 05:53:01 +04:00
|
|
|
* crypto_alloc_base() will first attempt to locate an already loaded
|
|
|
|
* algorithm. If that fails and the kernel supports dynamically loadable
|
|
|
|
* modules, it will then attempt to load a module of the same name or
|
|
|
|
* alias. If that fails it will send a query to any loaded crypto manager
|
|
|
|
* to construct an algorithm on the fly. A refcount is grabbed on the
|
|
|
|
* algorithm which is then associated with the new transform.
|
|
|
|
*
|
|
|
|
* The returned transform is of a non-determinate type. Most people
|
|
|
|
* should use one of the more specific allocation functions such as
|
crypto: skcipher - remove the "blkcipher" algorithm type
Now that all "blkcipher" algorithms have been converted to "skcipher",
remove the blkcipher algorithm type.
The skcipher (symmetric key cipher) algorithm type was introduced a few
years ago to replace both blkcipher and ablkcipher (synchronous and
asynchronous block cipher). The advantages of skcipher include:
- A much less confusing name, since none of these algorithm types have
ever actually been for raw block ciphers, but rather for all
length-preserving encryption modes including block cipher modes of
operation, stream ciphers, and other length-preserving modes.
- It unified blkcipher and ablkcipher into a single algorithm type
which supports both synchronous and asynchronous implementations.
Note, blkcipher already operated only on scatterlists, so the fact
that skcipher does too isn't a regression in functionality.
- Better type safety by using struct skcipher_alg, struct
crypto_skcipher, etc. instead of crypto_alg, crypto_tfm, etc.
- It sometimes simplifies the implementations of algorithms.
Also, the blkcipher API was no longer being tested.
Signed-off-by: Eric Biggers <ebiggers@google.com>
Signed-off-by: Herbert Xu <herbert@gondor.apana.org.au>
2019-10-25 22:41:12 +03:00
|
|
|
* crypto_alloc_skcipher().
|
2006-07-30 05:53:01 +04:00
|
|
|
*
|
|
|
|
* In case of error the return value is an error pointer.
|
|
|
|
*/
|
|
|
|
struct crypto_tfm *crypto_alloc_base(const char *alg_name, u32 type, u32 mask)
|
|
|
|
{
|
|
|
|
struct crypto_tfm *tfm;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
struct crypto_alg *alg;
|
|
|
|
|
|
|
|
alg = crypto_alg_mod_lookup(alg_name, type, mask);
|
2006-10-11 16:29:51 +04:00
|
|
|
if (IS_ERR(alg)) {
|
|
|
|
err = PTR_ERR(alg);
|
2006-07-30 05:53:01 +04:00
|
|
|
goto err;
|
2006-10-11 16:29:51 +04:00
|
|
|
}
|
2006-07-30 05:53:01 +04:00
|
|
|
|
2007-01-24 12:50:26 +03:00
|
|
|
tfm = __crypto_alloc_tfm(alg, type, mask);
|
2006-07-30 05:53:01 +04:00
|
|
|
if (!IS_ERR(tfm))
|
2006-10-11 16:29:51 +04:00
|
|
|
return tfm;
|
2006-07-30 05:53:01 +04:00
|
|
|
|
|
|
|
crypto_mod_put(alg);
|
|
|
|
err = PTR_ERR(tfm);
|
|
|
|
|
|
|
|
err:
|
|
|
|
if (err != -EAGAIN)
|
|
|
|
break;
|
2015-10-19 13:23:57 +03:00
|
|
|
if (fatal_signal_pending(current)) {
|
2006-07-30 05:53:01 +04:00
|
|
|
err = -EINTR;
|
|
|
|
break;
|
|
|
|
}
|
2006-10-11 16:29:51 +04:00
|
|
|
}
|
2006-07-30 05:53:01 +04:00
|
|
|
|
2006-10-11 16:29:51 +04:00
|
|
|
return ERR_PTR(err);
|
2006-07-30 05:53:01 +04:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(crypto_alloc_base);
|
2008-09-21 01:52:53 +04:00
|
|
|
|
2009-02-18 11:56:59 +03:00
|
|
|
void *crypto_create_tfm(struct crypto_alg *alg,
|
|
|
|
const struct crypto_type *frontend)
|
2008-09-21 01:52:53 +04:00
|
|
|
{
|
|
|
|
char *mem;
|
|
|
|
struct crypto_tfm *tfm = NULL;
|
|
|
|
unsigned int tfmsize;
|
|
|
|
unsigned int total;
|
|
|
|
int err = -ENOMEM;
|
|
|
|
|
|
|
|
tfmsize = frontend->tfmsize;
|
2009-07-13 16:46:25 +04:00
|
|
|
total = tfmsize + sizeof(*tfm) + frontend->extsize(alg);
|
2008-09-21 01:52:53 +04:00
|
|
|
|
|
|
|
mem = kzalloc(total, GFP_KERNEL);
|
|
|
|
if (mem == NULL)
|
|
|
|
goto out_err;
|
|
|
|
|
|
|
|
tfm = (struct crypto_tfm *)(mem + tfmsize);
|
|
|
|
tfm->__crt_alg = alg;
|
|
|
|
|
2009-07-13 16:46:25 +04:00
|
|
|
err = frontend->init_tfm(tfm);
|
2008-09-21 01:52:53 +04:00
|
|
|
if (err)
|
|
|
|
goto out_free_tfm;
|
|
|
|
|
|
|
|
if (!tfm->exit && alg->cra_init && (err = alg->cra_init(tfm)))
|
|
|
|
goto cra_init_failed;
|
|
|
|
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
cra_init_failed:
|
|
|
|
crypto_exit_ops(tfm);
|
|
|
|
out_free_tfm:
|
|
|
|
if (err == -EAGAIN)
|
|
|
|
crypto_shoot_alg(alg);
|
|
|
|
kfree(mem);
|
|
|
|
out_err:
|
2009-02-18 11:56:59 +03:00
|
|
|
mem = ERR_PTR(err);
|
2008-09-21 01:52:53 +04:00
|
|
|
out:
|
2009-02-18 11:56:59 +03:00
|
|
|
return mem;
|
2008-09-21 01:52:53 +04:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(crypto_create_tfm);
|
|
|
|
|
2009-07-08 13:53:16 +04:00
|
|
|
struct crypto_alg *crypto_find_alg(const char *alg_name,
|
|
|
|
const struct crypto_type *frontend,
|
|
|
|
u32 type, u32 mask)
|
|
|
|
{
|
|
|
|
if (frontend) {
|
|
|
|
type &= frontend->maskclear;
|
|
|
|
mask &= frontend->maskclear;
|
|
|
|
type |= frontend->type;
|
|
|
|
mask |= frontend->maskset;
|
|
|
|
}
|
|
|
|
|
2018-03-20 02:41:00 +03:00
|
|
|
return crypto_alg_mod_lookup(alg_name, type, mask);
|
2009-07-08 13:53:16 +04:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(crypto_find_alg);
|
|
|
|
|
2008-09-21 01:52:53 +04:00
|
|
|
/*
|
|
|
|
* crypto_alloc_tfm - Locate algorithm and allocate transform
|
|
|
|
* @alg_name: Name of algorithm
|
|
|
|
* @frontend: Frontend algorithm type
|
|
|
|
* @type: Type of algorithm
|
|
|
|
* @mask: Mask for type comparison
|
|
|
|
*
|
|
|
|
* crypto_alloc_tfm() will first attempt to locate an already loaded
|
|
|
|
* algorithm. If that fails and the kernel supports dynamically loadable
|
|
|
|
* modules, it will then attempt to load a module of the same name or
|
|
|
|
* alias. If that fails it will send a query to any loaded crypto manager
|
|
|
|
* to construct an algorithm on the fly. A refcount is grabbed on the
|
|
|
|
* algorithm which is then associated with the new transform.
|
|
|
|
*
|
|
|
|
* The returned transform is of a non-determinate type. Most people
|
|
|
|
* should use one of the more specific allocation functions such as
|
2019-11-29 21:16:48 +03:00
|
|
|
* crypto_alloc_skcipher().
|
2008-09-21 01:52:53 +04:00
|
|
|
*
|
|
|
|
* In case of error the return value is an error pointer.
|
|
|
|
*/
|
2009-02-18 11:56:59 +03:00
|
|
|
void *crypto_alloc_tfm(const char *alg_name,
|
|
|
|
const struct crypto_type *frontend, u32 type, u32 mask)
|
2008-09-21 01:52:53 +04:00
|
|
|
{
|
2009-02-18 11:56:59 +03:00
|
|
|
void *tfm;
|
2008-09-21 01:52:53 +04:00
|
|
|
int err;
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
struct crypto_alg *alg;
|
|
|
|
|
2009-07-08 13:53:16 +04:00
|
|
|
alg = crypto_find_alg(alg_name, frontend, type, mask);
|
2008-09-21 01:52:53 +04:00
|
|
|
if (IS_ERR(alg)) {
|
|
|
|
err = PTR_ERR(alg);
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
|
|
|
tfm = crypto_create_tfm(alg, frontend);
|
|
|
|
if (!IS_ERR(tfm))
|
|
|
|
return tfm;
|
|
|
|
|
|
|
|
crypto_mod_put(alg);
|
|
|
|
err = PTR_ERR(tfm);
|
|
|
|
|
|
|
|
err:
|
|
|
|
if (err != -EAGAIN)
|
|
|
|
break;
|
2015-10-19 13:23:57 +03:00
|
|
|
if (fatal_signal_pending(current)) {
|
2008-09-21 01:52:53 +04:00
|
|
|
err = -EINTR;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ERR_PTR(err);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(crypto_alloc_tfm);
|
2009-02-05 08:48:24 +03:00
|
|
|
|
2006-07-30 05:53:01 +04:00
|
|
|
/*
|
2009-02-05 08:48:24 +03:00
|
|
|
* crypto_destroy_tfm - Free crypto transform
|
|
|
|
* @mem: Start of tfm slab
|
2006-07-30 05:53:01 +04:00
|
|
|
* @tfm: Transform to free
|
|
|
|
*
|
2009-02-05 08:48:24 +03:00
|
|
|
* This function frees up the transform and any associated resources,
|
2006-07-30 05:53:01 +04:00
|
|
|
* then drops the refcount on the associated algorithm.
|
|
|
|
*/
|
2009-02-05 08:48:24 +03:00
|
|
|
void crypto_destroy_tfm(void *mem, struct crypto_tfm *tfm)
|
2005-04-17 02:20:36 +04:00
|
|
|
{
|
2005-07-07 00:54:31 +04:00
|
|
|
struct crypto_alg *alg;
|
|
|
|
|
2009-02-05 08:48:24 +03:00
|
|
|
if (unlikely(!mem))
|
2005-07-07 00:54:31 +04:00
|
|
|
return;
|
|
|
|
|
|
|
|
alg = tfm->__crt_alg;
|
2005-04-17 02:20:36 +04:00
|
|
|
|
2008-09-14 05:19:03 +04:00
|
|
|
if (!tfm->exit && alg->cra_exit)
|
2006-05-24 07:02:26 +04:00
|
|
|
alg->cra_exit(tfm);
|
2005-04-17 02:20:36 +04:00
|
|
|
crypto_exit_ops(tfm);
|
2006-05-28 03:05:24 +04:00
|
|
|
crypto_mod_put(alg);
|
2009-03-29 11:20:48 +04:00
|
|
|
kzfree(mem);
|
2005-04-17 02:20:36 +04:00
|
|
|
}
|
2009-02-05 08:48:24 +03:00
|
|
|
EXPORT_SYMBOL_GPL(crypto_destroy_tfm);
|
2006-08-26 11:35:45 +04:00
|
|
|
|
|
|
|
int crypto_has_alg(const char *name, u32 type, u32 mask)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
struct crypto_alg *alg = crypto_alg_mod_lookup(name, type, mask);
|
2010-02-16 15:26:46 +03:00
|
|
|
|
2006-08-26 11:35:45 +04:00
|
|
|
if (!IS_ERR(alg)) {
|
|
|
|
crypto_mod_put(alg);
|
|
|
|
ret = 1;
|
|
|
|
}
|
2010-02-16 15:26:46 +03:00
|
|
|
|
2006-08-26 11:35:45 +04:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(crypto_has_alg);
|
2008-03-30 12:36:09 +04:00
|
|
|
|
2017-10-18 10:00:38 +03:00
|
|
|
void crypto_req_done(struct crypto_async_request *req, int err)
|
|
|
|
{
|
|
|
|
struct crypto_wait *wait = req->data;
|
|
|
|
|
|
|
|
if (err == -EINPROGRESS)
|
|
|
|
return;
|
|
|
|
|
|
|
|
wait->err = err;
|
|
|
|
complete(&wait->completion);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(crypto_req_done);
|
|
|
|
|
2008-03-30 12:36:09 +04:00
|
|
|
MODULE_DESCRIPTION("Cryptographic core API");
|
|
|
|
MODULE_LICENSE("GPL");
|
2019-11-08 13:26:30 +03:00
|
|
|
MODULE_SOFTDEP("pre: cryptomgr");
|