Commit 317b6ab8 authored by Alexander Shishkin's avatar Alexander Shishkin Committed by Stefan Bader

stm class: Fix locking in unbinding policy path

BugLink: https://bugs.launchpad.net/bugs/1826212

[ Upstream commit 4c127fd1 ]

Right now, if stm device removal has to unbind from a policy (that is,
an stm device that has STP policy, gets removed), it will trigger a
nested lock on the stm device's policy mutex.

This patch fixes the problem by moving the locking from the policy
unbinding to policy removal (configfs path), where it's actually needed;
the other caller of the policy unbinding function already takes the
mutex around the call.
Signed-off-by: default avatarAlexander Shishkin <alexander.shishkin@linux.intel.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Signed-off-by: default avatarSasha Levin <sashal@kernel.org>
Signed-off-by: default avatarStefan Bader <stefan.bader@canonical.com>
Signed-off-by: default avatarKleber Sacilotto de Souza <kleber.souza@canonical.com>
parent 3375e686
...@@ -272,13 +272,17 @@ void stp_policy_unbind(struct stp_policy *policy) ...@@ -272,13 +272,17 @@ void stp_policy_unbind(struct stp_policy *policy)
{ {
struct stm_device *stm = policy->stm; struct stm_device *stm = policy->stm;
/*
* stp_policy_release() will not call here if the policy is already
* unbound; other users should not either, as no link exists between
* this policy and anything else in that case
*/
if (WARN_ON_ONCE(!policy->stm)) if (WARN_ON_ONCE(!policy->stm))
return; return;
mutex_lock(&stm->policy_mutex); lockdep_assert_held(&stm->policy_mutex);
stm->policy = NULL;
mutex_unlock(&stm->policy_mutex);
stm->policy = NULL;
policy->stm = NULL; policy->stm = NULL;
stm_put_device(stm); stm_put_device(stm);
...@@ -287,8 +291,16 @@ void stp_policy_unbind(struct stp_policy *policy) ...@@ -287,8 +291,16 @@ void stp_policy_unbind(struct stp_policy *policy)
static void stp_policy_release(struct config_item *item) static void stp_policy_release(struct config_item *item)
{ {
struct stp_policy *policy = to_stp_policy(item); struct stp_policy *policy = to_stp_policy(item);
struct stm_device *stm = policy->stm;
/* a policy *can* be unbound and still exist in configfs tree */
if (!stm)
return;
mutex_lock(&stm->policy_mutex);
stp_policy_unbind(policy); stp_policy_unbind(policy);
mutex_unlock(&stm->policy_mutex);
kfree(policy); kfree(policy);
} }
......
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