diff options
author | Steffen Klassert <steffen.klassert@secunet.com> | 2011-11-08 04:09:17 -0500 |
---|---|---|
committer | Herbert Xu <herbert@gondor.apana.org.au> | 2011-11-08 23:04:06 -0500 |
commit | ce3fd840f588d85a8c1be651cf90fa1ba1f029e9 (patch) | |
tree | d113057ca46784cfa34380ab44f73dde76bda1fd | |
parent | 505172e11f5a0d9916e20e40d3b0a6f87d3a59b6 (diff) |
crypto: Unlink and free instances when deleted
We leak the crypto instance when we unregister an instance with
crypto_del_alg(). Therefore we introduce crypto_unregister_instance()
to unlink the crypto instance from the template's instances list and
to free the recources of the instance properly.
Signed-off-by: Steffen Klassert <steffen.klassert@secunet.com>
Signed-off-by: Herbert Xu <herbert@gondor.apana.org.au>
-rw-r--r-- | crypto/algapi.c | 29 | ||||
-rw-r--r-- | crypto/crypto_user.c | 2 | ||||
-rw-r--r-- | include/crypto/algapi.h | 1 |
3 files changed, 31 insertions, 1 deletions
diff --git a/crypto/algapi.c b/crypto/algapi.c index 54dd4e33b5d6..9d4a9fe913f8 100644 --- a/crypto/algapi.c +++ b/crypto/algapi.c | |||
@@ -518,6 +518,35 @@ err: | |||
518 | } | 518 | } |
519 | EXPORT_SYMBOL_GPL(crypto_register_instance); | 519 | EXPORT_SYMBOL_GPL(crypto_register_instance); |
520 | 520 | ||
521 | int crypto_unregister_instance(struct crypto_alg *alg) | ||
522 | { | ||
523 | int err; | ||
524 | struct crypto_instance *inst = (void *)alg; | ||
525 | struct crypto_template *tmpl = inst->tmpl; | ||
526 | LIST_HEAD(users); | ||
527 | |||
528 | if (!(alg->cra_flags & CRYPTO_ALG_INSTANCE)) | ||
529 | return -EINVAL; | ||
530 | |||
531 | BUG_ON(atomic_read(&alg->cra_refcnt) != 1); | ||
532 | |||
533 | down_write(&crypto_alg_sem); | ||
534 | |||
535 | hlist_del_init(&inst->list); | ||
536 | err = crypto_remove_alg(alg, &users); | ||
537 | |||
538 | up_write(&crypto_alg_sem); | ||
539 | |||
540 | if (err) | ||
541 | return err; | ||
542 | |||
543 | tmpl->free(inst); | ||
544 | crypto_remove_final(&users); | ||
545 | |||
546 | return 0; | ||
547 | } | ||
548 | EXPORT_SYMBOL_GPL(crypto_unregister_instance); | ||
549 | |||
521 | int crypto_init_spawn(struct crypto_spawn *spawn, struct crypto_alg *alg, | 550 | int crypto_init_spawn(struct crypto_spawn *spawn, struct crypto_alg *alg, |
522 | struct crypto_instance *inst, u32 mask) | 551 | struct crypto_instance *inst, u32 mask) |
523 | { | 552 | { |
diff --git a/crypto/crypto_user.c b/crypto/crypto_user.c index 2abca780312d..edeebe1a84bb 100644 --- a/crypto/crypto_user.c +++ b/crypto/crypto_user.c | |||
@@ -301,7 +301,7 @@ static int crypto_del_alg(struct sk_buff *skb, struct nlmsghdr *nlh, | |||
301 | if (atomic_read(&alg->cra_refcnt) != 1) | 301 | if (atomic_read(&alg->cra_refcnt) != 1) |
302 | return -EBUSY; | 302 | return -EBUSY; |
303 | 303 | ||
304 | return crypto_unregister_alg(alg); | 304 | return crypto_unregister_instance(alg); |
305 | } | 305 | } |
306 | 306 | ||
307 | static int crypto_add_alg(struct sk_buff *skb, struct nlmsghdr *nlh, | 307 | static int crypto_add_alg(struct sk_buff *skb, struct nlmsghdr *nlh, |
diff --git a/include/crypto/algapi.h b/include/crypto/algapi.h index ecc721def10c..418d270e1806 100644 --- a/include/crypto/algapi.h +++ b/include/crypto/algapi.h | |||
@@ -134,6 +134,7 @@ struct crypto_template *crypto_lookup_template(const char *name); | |||
134 | 134 | ||
135 | int crypto_register_instance(struct crypto_template *tmpl, | 135 | int crypto_register_instance(struct crypto_template *tmpl, |
136 | struct crypto_instance *inst); | 136 | struct crypto_instance *inst); |
137 | int crypto_unregister_instance(struct crypto_alg *alg); | ||
137 | 138 | ||
138 | int crypto_init_spawn(struct crypto_spawn *spawn, struct crypto_alg *alg, | 139 | int crypto_init_spawn(struct crypto_spawn *spawn, struct crypto_alg *alg, |
139 | struct crypto_instance *inst, u32 mask); | 140 | struct crypto_instance *inst, u32 mask); |