Commit cf1fac94 authored by Viresh Kumar's avatar Viresh Kumar

opp: Reduce the size of critical section in _opp_kref_release()

There is a lot of stuff here which can be done outside of the
opp_table->lock, do that. This helps avoiding a circular dependency
lockdeps around debugfs.
Reported-by: default avatarRob Clark <robdclark@gmail.com>
Signed-off-by: default avatarViresh Kumar <viresh.kumar@linaro.org>
parent 9e62edac
...@@ -1252,9 +1252,14 @@ void _opp_free(struct dev_pm_opp *opp) ...@@ -1252,9 +1252,14 @@ void _opp_free(struct dev_pm_opp *opp)
kfree(opp); kfree(opp);
} }
static void _opp_kref_release(struct dev_pm_opp *opp, static void _opp_kref_release(struct kref *kref)
struct opp_table *opp_table)
{ {
struct dev_pm_opp *opp = container_of(kref, struct dev_pm_opp, kref);
struct opp_table *opp_table = opp->opp_table;
list_del(&opp->node);
mutex_unlock(&opp_table->lock);
/* /*
* Notify the changes in the availability of the operable * Notify the changes in the availability of the operable
* frequency/voltage list. * frequency/voltage list.
...@@ -1262,27 +1267,9 @@ static void _opp_kref_release(struct dev_pm_opp *opp, ...@@ -1262,27 +1267,9 @@ static void _opp_kref_release(struct dev_pm_opp *opp,
blocking_notifier_call_chain(&opp_table->head, OPP_EVENT_REMOVE, opp); blocking_notifier_call_chain(&opp_table->head, OPP_EVENT_REMOVE, opp);
_of_opp_free_required_opps(opp_table, opp); _of_opp_free_required_opps(opp_table, opp);
opp_debug_remove_one(opp); opp_debug_remove_one(opp);
list_del(&opp->node);
kfree(opp); kfree(opp);
} }
static void _opp_kref_release_unlocked(struct kref *kref)
{
struct dev_pm_opp *opp = container_of(kref, struct dev_pm_opp, kref);
struct opp_table *opp_table = opp->opp_table;
_opp_kref_release(opp, opp_table);
}
static void _opp_kref_release_locked(struct kref *kref)
{
struct dev_pm_opp *opp = container_of(kref, struct dev_pm_opp, kref);
struct opp_table *opp_table = opp->opp_table;
_opp_kref_release(opp, opp_table);
mutex_unlock(&opp_table->lock);
}
void dev_pm_opp_get(struct dev_pm_opp *opp) void dev_pm_opp_get(struct dev_pm_opp *opp)
{ {
kref_get(&opp->kref); kref_get(&opp->kref);
...@@ -1290,16 +1277,10 @@ void dev_pm_opp_get(struct dev_pm_opp *opp) ...@@ -1290,16 +1277,10 @@ void dev_pm_opp_get(struct dev_pm_opp *opp)
void dev_pm_opp_put(struct dev_pm_opp *opp) void dev_pm_opp_put(struct dev_pm_opp *opp)
{ {
kref_put_mutex(&opp->kref, _opp_kref_release_locked, kref_put_mutex(&opp->kref, _opp_kref_release, &opp->opp_table->lock);
&opp->opp_table->lock);
} }
EXPORT_SYMBOL_GPL(dev_pm_opp_put); EXPORT_SYMBOL_GPL(dev_pm_opp_put);
static void dev_pm_opp_put_unlocked(struct dev_pm_opp *opp)
{
kref_put(&opp->kref, _opp_kref_release_unlocked);
}
/** /**
* dev_pm_opp_remove() - Remove an OPP from OPP table * dev_pm_opp_remove() - Remove an OPP from OPP table
* @dev: device for which we do this operation * @dev: device for which we do this operation
...@@ -1343,30 +1324,49 @@ void dev_pm_opp_remove(struct device *dev, unsigned long freq) ...@@ -1343,30 +1324,49 @@ void dev_pm_opp_remove(struct device *dev, unsigned long freq)
} }
EXPORT_SYMBOL_GPL(dev_pm_opp_remove); EXPORT_SYMBOL_GPL(dev_pm_opp_remove);
static struct dev_pm_opp *_opp_get_next(struct opp_table *opp_table,
bool dynamic)
{
struct dev_pm_opp *opp = NULL, *temp;
mutex_lock(&opp_table->lock);
list_for_each_entry(temp, &opp_table->opp_list, node) {
if (dynamic == temp->dynamic) {
opp = temp;
break;
}
}
mutex_unlock(&opp_table->lock);
return opp;
}
bool _opp_remove_all_static(struct opp_table *opp_table) bool _opp_remove_all_static(struct opp_table *opp_table)
{ {
struct dev_pm_opp *opp, *tmp; struct dev_pm_opp *opp;
bool ret = true;
mutex_lock(&opp_table->lock); mutex_lock(&opp_table->lock);
if (!opp_table->parsed_static_opps) { if (!opp_table->parsed_static_opps) {
ret = false; mutex_unlock(&opp_table->lock);
goto unlock; return false;
} }
if (--opp_table->parsed_static_opps) if (--opp_table->parsed_static_opps) {
goto unlock; mutex_unlock(&opp_table->lock);
return true;
list_for_each_entry_safe(opp, tmp, &opp_table->opp_list, node) {
if (!opp->dynamic)
dev_pm_opp_put_unlocked(opp);
} }
unlock:
mutex_unlock(&opp_table->lock); mutex_unlock(&opp_table->lock);
return ret; /*
* Can't remove the OPP from under the lock, debugfs removal needs to
* happen lock less to avoid circular dependency issues.
*/
while ((opp = _opp_get_next(opp_table, false)))
dev_pm_opp_put(opp);
return true;
} }
/** /**
...@@ -1378,21 +1378,21 @@ bool _opp_remove_all_static(struct opp_table *opp_table) ...@@ -1378,21 +1378,21 @@ bool _opp_remove_all_static(struct opp_table *opp_table)
void dev_pm_opp_remove_all_dynamic(struct device *dev) void dev_pm_opp_remove_all_dynamic(struct device *dev)
{ {
struct opp_table *opp_table; struct opp_table *opp_table;
struct dev_pm_opp *opp, *temp; struct dev_pm_opp *opp;
int count = 0; int count = 0;
opp_table = _find_opp_table(dev); opp_table = _find_opp_table(dev);
if (IS_ERR(opp_table)) if (IS_ERR(opp_table))
return; return;
mutex_lock(&opp_table->lock); /*
list_for_each_entry_safe(opp, temp, &opp_table->opp_list, node) { * Can't remove the OPP from under the lock, debugfs removal needs to
if (opp->dynamic) { * happen lock less to avoid circular dependency issues.
dev_pm_opp_put_unlocked(opp); */
count++; while ((opp = _opp_get_next(opp_table, true))) {
} dev_pm_opp_put(opp);
count++;
} }
mutex_unlock(&opp_table->lock);
/* Drop the references taken by dev_pm_opp_add() */ /* Drop the references taken by dev_pm_opp_add() */
while (count--) while (count--)
......
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