Commit 5f567fff authored by Herbert Xu's avatar Herbert Xu

crypto: api - Retain alg refcount in crypto_grab_spawn

This patch changes crypto_grab_spawn to retain the reference count
on the algorithm.  This is because the caller needs to access the
algorithm parameters and without the reference count the algorithm
can be freed at any time.

The reference count will be subsequently dropped by the crypto API
once the instance has been registered.  The helper crypto_drop_spawn
will also conditionally drop the reference count depending on whether
it has been registered.

Note that the code is actually added to crypto_init_spawn.  However,
unless the caller activates this by setting spawn->dropref beforehand
then nothing happens.  The only caller that sets dropref is currently
crypto_grab_spawn.

Once all legacy users of crypto_init_spawn disappear, then we can
kill the dropref flag.

Internally each instance will maintain a list of its spawns prior
to registration.  This memory used by this list is shared with
other fields that are only used after registration.  In order for
this to work a new flag spawn->registered is added to indicate
whether spawn->inst can be used.

Fixes: d6ef2f19 ("crypto: api - Add crypto_grab_spawn primitive")
Signed-off-by: default avatarHerbert Xu <herbert@gondor.apana.org.au>
parent 3932aa1c
...@@ -124,8 +124,6 @@ static void crypto_remove_instance(struct crypto_instance *inst, ...@@ -124,8 +124,6 @@ static void crypto_remove_instance(struct crypto_instance *inst,
return; return;
inst->alg.cra_flags |= CRYPTO_ALG_DEAD; inst->alg.cra_flags |= CRYPTO_ALG_DEAD;
if (hlist_unhashed(&inst->list))
return;
if (!tmpl || !crypto_tmpl_get(tmpl)) if (!tmpl || !crypto_tmpl_get(tmpl))
return; return;
...@@ -175,17 +173,26 @@ void crypto_remove_spawns(struct crypto_alg *alg, struct list_head *list, ...@@ -175,17 +173,26 @@ void crypto_remove_spawns(struct crypto_alg *alg, struct list_head *list,
list); list);
inst = spawn->inst; inst = spawn->inst;
BUG_ON(&inst->alg == alg);
list_move(&spawn->list, &stack); list_move(&spawn->list, &stack);
spawn->dead = !spawn->registered || &inst->alg != nalg;
if (!spawn->registered)
break;
BUG_ON(&inst->alg == alg);
if (&inst->alg == nalg) if (&inst->alg == nalg)
break; break;
spawn->dead = true;
spawns = &inst->alg.cra_users; spawns = &inst->alg.cra_users;
/* /*
* Even if spawn->registered is true, the
* instance itself may still be unregistered.
* This is because it may have failed during
* registration. Therefore we still need to
* make the following test.
*
* We may encounter an unregistered instance here, since * We may encounter an unregistered instance here, since
* an instance's spawns are set up prior to the instance * an instance's spawns are set up prior to the instance
* being registered. An unregistered instance will have * being registered. An unregistered instance will have
...@@ -208,7 +215,7 @@ void crypto_remove_spawns(struct crypto_alg *alg, struct list_head *list, ...@@ -208,7 +215,7 @@ void crypto_remove_spawns(struct crypto_alg *alg, struct list_head *list,
list_for_each_entry_safe(spawn, n, &secondary_spawns, list) { list_for_each_entry_safe(spawn, n, &secondary_spawns, list) {
if (!spawn->dead) if (!spawn->dead)
list_move(&spawn->list, &spawn->alg->cra_users); list_move(&spawn->list, &spawn->alg->cra_users);
else else if (spawn->registered)
crypto_remove_instance(spawn->inst, list); crypto_remove_instance(spawn->inst, list);
} }
} }
...@@ -599,6 +606,7 @@ int crypto_register_instance(struct crypto_template *tmpl, ...@@ -599,6 +606,7 @@ int crypto_register_instance(struct crypto_template *tmpl,
struct crypto_instance *inst) struct crypto_instance *inst)
{ {
struct crypto_larval *larval; struct crypto_larval *larval;
struct crypto_spawn *spawn;
int err; int err;
err = crypto_check_alg(&inst->alg); err = crypto_check_alg(&inst->alg);
...@@ -610,6 +618,23 @@ int crypto_register_instance(struct crypto_template *tmpl, ...@@ -610,6 +618,23 @@ int crypto_register_instance(struct crypto_template *tmpl,
down_write(&crypto_alg_sem); down_write(&crypto_alg_sem);
larval = ERR_PTR(-EAGAIN);
for (spawn = inst->spawns; spawn;) {
struct crypto_spawn *next;
if (spawn->dead)
goto unlock;
next = spawn->next;
spawn->inst = inst;
spawn->registered = true;
if (spawn->dropref)
crypto_mod_put(spawn->alg);
spawn = next;
}
larval = __crypto_register_alg(&inst->alg); larval = __crypto_register_alg(&inst->alg);
if (IS_ERR(larval)) if (IS_ERR(larval))
goto unlock; goto unlock;
...@@ -655,7 +680,9 @@ int crypto_init_spawn(struct crypto_spawn *spawn, struct crypto_alg *alg, ...@@ -655,7 +680,9 @@ int crypto_init_spawn(struct crypto_spawn *spawn, struct crypto_alg *alg,
if (WARN_ON_ONCE(inst == NULL)) if (WARN_ON_ONCE(inst == NULL))
return -EINVAL; return -EINVAL;
spawn->inst = inst; spawn->next = inst->spawns;
inst->spawns = spawn;
spawn->mask = mask; spawn->mask = mask;
down_write(&crypto_alg_sem); down_write(&crypto_alg_sem);
...@@ -697,8 +724,10 @@ int crypto_grab_spawn(struct crypto_spawn *spawn, const char *name, ...@@ -697,8 +724,10 @@ int crypto_grab_spawn(struct crypto_spawn *spawn, const char *name,
if (IS_ERR(alg)) if (IS_ERR(alg))
return PTR_ERR(alg); return PTR_ERR(alg);
spawn->dropref = true;
err = crypto_init_spawn(spawn, alg, spawn->inst, mask); err = crypto_init_spawn(spawn, alg, spawn->inst, mask);
crypto_mod_put(alg); if (err)
crypto_mod_put(alg);
return err; return err;
} }
EXPORT_SYMBOL_GPL(crypto_grab_spawn); EXPORT_SYMBOL_GPL(crypto_grab_spawn);
...@@ -709,6 +738,9 @@ void crypto_drop_spawn(struct crypto_spawn *spawn) ...@@ -709,6 +738,9 @@ void crypto_drop_spawn(struct crypto_spawn *spawn)
if (!spawn->dead) if (!spawn->dead)
list_del(&spawn->list); list_del(&spawn->list);
up_write(&crypto_alg_sem); up_write(&crypto_alg_sem);
if (spawn->dropref && !spawn->registered)
crypto_mod_put(spawn->alg);
} }
EXPORT_SYMBOL_GPL(crypto_drop_spawn); EXPORT_SYMBOL_GPL(crypto_drop_spawn);
......
...@@ -47,7 +47,13 @@ struct crypto_instance { ...@@ -47,7 +47,13 @@ struct crypto_instance {
struct crypto_alg alg; struct crypto_alg alg;
struct crypto_template *tmpl; struct crypto_template *tmpl;
struct hlist_node list;
union {
/* Node in list of instances after registration. */
struct hlist_node list;
/* List of attached spawns before registration. */
struct crypto_spawn *spawns;
};
void *__ctx[] CRYPTO_MINALIGN_ATTR; void *__ctx[] CRYPTO_MINALIGN_ATTR;
}; };
...@@ -67,10 +73,17 @@ struct crypto_template { ...@@ -67,10 +73,17 @@ struct crypto_template {
struct crypto_spawn { struct crypto_spawn {
struct list_head list; struct list_head list;
struct crypto_alg *alg; struct crypto_alg *alg;
struct crypto_instance *inst; union {
/* Back pointer to instance after registration.*/
struct crypto_instance *inst;
/* Spawn list pointer prior to registration. */
struct crypto_spawn *next;
};
const struct crypto_type *frontend; const struct crypto_type *frontend;
u32 mask; u32 mask;
bool dead; bool dead;
bool dropref;
bool registered;
}; };
struct crypto_queue { struct crypto_queue {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment