Commit 35241d12 authored by Lina Iyer's avatar Lina Iyer Committed by Rafael J. Wysocki

PM / Domains: Abstract genpd locking

Abstract genpd lock/unlock calls, in preparation for domain specific
locks added in the following patches.
Signed-off-by: default avatarLina Iyer <lina.iyer@linaro.org>
Signed-off-by: default avatarUlf Hansson <ulf.hansson@linaro.org>
Reviewed-by: default avatarKevin Hilman <khilman@baylibre.com>
Signed-off-by: default avatarRafael J. Wysocki <rafael.j.wysocki@intel.com>
parent 70bb510e
...@@ -39,6 +39,46 @@ ...@@ -39,6 +39,46 @@
static LIST_HEAD(gpd_list); static LIST_HEAD(gpd_list);
static DEFINE_MUTEX(gpd_list_lock); static DEFINE_MUTEX(gpd_list_lock);
struct genpd_lock_ops {
void (*lock)(struct generic_pm_domain *genpd);
void (*lock_nested)(struct generic_pm_domain *genpd, int depth);
int (*lock_interruptible)(struct generic_pm_domain *genpd);
void (*unlock)(struct generic_pm_domain *genpd);
};
static void genpd_lock_mtx(struct generic_pm_domain *genpd)
{
mutex_lock(&genpd->mlock);
}
static void genpd_lock_nested_mtx(struct generic_pm_domain *genpd,
int depth)
{
mutex_lock_nested(&genpd->mlock, depth);
}
static int genpd_lock_interruptible_mtx(struct generic_pm_domain *genpd)
{
return mutex_lock_interruptible(&genpd->mlock);
}
static void genpd_unlock_mtx(struct generic_pm_domain *genpd)
{
return mutex_unlock(&genpd->mlock);
}
static const struct genpd_lock_ops genpd_mtx_ops = {
.lock = genpd_lock_mtx,
.lock_nested = genpd_lock_nested_mtx,
.lock_interruptible = genpd_lock_interruptible_mtx,
.unlock = genpd_unlock_mtx,
};
#define genpd_lock(p) p->lock_ops->lock(p)
#define genpd_lock_nested(p, d) p->lock_ops->lock_nested(p, d)
#define genpd_lock_interruptible(p) p->lock_ops->lock_interruptible(p)
#define genpd_unlock(p) p->lock_ops->unlock(p)
/* /*
* Get the generic PM domain for a particular struct device. * Get the generic PM domain for a particular struct device.
* This validates the struct device pointer, the PM domain pointer, * This validates the struct device pointer, the PM domain pointer,
...@@ -200,9 +240,9 @@ static int genpd_poweron(struct generic_pm_domain *genpd, unsigned int depth) ...@@ -200,9 +240,9 @@ static int genpd_poweron(struct generic_pm_domain *genpd, unsigned int depth)
genpd_sd_counter_inc(master); genpd_sd_counter_inc(master);
mutex_lock_nested(&master->lock, depth + 1); genpd_lock_nested(master, depth + 1);
ret = genpd_poweron(master, depth + 1); ret = genpd_poweron(master, depth + 1);
mutex_unlock(&master->lock); genpd_unlock(master);
if (ret) { if (ret) {
genpd_sd_counter_dec(master); genpd_sd_counter_dec(master);
...@@ -255,9 +295,9 @@ static int genpd_dev_pm_qos_notifier(struct notifier_block *nb, ...@@ -255,9 +295,9 @@ static int genpd_dev_pm_qos_notifier(struct notifier_block *nb,
spin_unlock_irq(&dev->power.lock); spin_unlock_irq(&dev->power.lock);
if (!IS_ERR(genpd)) { if (!IS_ERR(genpd)) {
mutex_lock(&genpd->lock); genpd_lock(genpd);
genpd->max_off_time_changed = true; genpd->max_off_time_changed = true;
mutex_unlock(&genpd->lock); genpd_unlock(genpd);
} }
dev = dev->parent; dev = dev->parent;
...@@ -354,9 +394,9 @@ static void genpd_power_off_work_fn(struct work_struct *work) ...@@ -354,9 +394,9 @@ static void genpd_power_off_work_fn(struct work_struct *work)
genpd = container_of(work, struct generic_pm_domain, power_off_work); genpd = container_of(work, struct generic_pm_domain, power_off_work);
mutex_lock(&genpd->lock); genpd_lock(genpd);
genpd_poweroff(genpd, true); genpd_poweroff(genpd, true);
mutex_unlock(&genpd->lock); genpd_unlock(genpd);
} }
/** /**
...@@ -472,9 +512,9 @@ static int genpd_runtime_suspend(struct device *dev) ...@@ -472,9 +512,9 @@ static int genpd_runtime_suspend(struct device *dev)
if (dev->power.irq_safe) if (dev->power.irq_safe)
return 0; return 0;
mutex_lock(&genpd->lock); genpd_lock(genpd);
genpd_poweroff(genpd, false); genpd_poweroff(genpd, false);
mutex_unlock(&genpd->lock); genpd_unlock(genpd);
return 0; return 0;
} }
...@@ -509,9 +549,9 @@ static int genpd_runtime_resume(struct device *dev) ...@@ -509,9 +549,9 @@ static int genpd_runtime_resume(struct device *dev)
goto out; goto out;
} }
mutex_lock(&genpd->lock); genpd_lock(genpd);
ret = genpd_poweron(genpd, 0); ret = genpd_poweron(genpd, 0);
mutex_unlock(&genpd->lock); genpd_unlock(genpd);
if (ret) if (ret)
return ret; return ret;
...@@ -547,9 +587,9 @@ static int genpd_runtime_resume(struct device *dev) ...@@ -547,9 +587,9 @@ static int genpd_runtime_resume(struct device *dev)
genpd_stop_dev(genpd, dev); genpd_stop_dev(genpd, dev);
err_poweroff: err_poweroff:
if (!dev->power.irq_safe) { if (!dev->power.irq_safe) {
mutex_lock(&genpd->lock); genpd_lock(genpd);
genpd_poweroff(genpd, 0); genpd_poweroff(genpd, 0);
mutex_unlock(&genpd->lock); genpd_unlock(genpd);
} }
return ret; return ret;
...@@ -732,20 +772,20 @@ static int pm_genpd_prepare(struct device *dev) ...@@ -732,20 +772,20 @@ static int pm_genpd_prepare(struct device *dev)
if (resume_needed(dev, genpd)) if (resume_needed(dev, genpd))
pm_runtime_resume(dev); pm_runtime_resume(dev);
mutex_lock(&genpd->lock); genpd_lock(genpd);
if (genpd->prepared_count++ == 0) if (genpd->prepared_count++ == 0)
genpd->suspended_count = 0; genpd->suspended_count = 0;
mutex_unlock(&genpd->lock); genpd_unlock(genpd);
ret = pm_generic_prepare(dev); ret = pm_generic_prepare(dev);
if (ret) { if (ret) {
mutex_lock(&genpd->lock); genpd_lock(genpd);
genpd->prepared_count--; genpd->prepared_count--;
mutex_unlock(&genpd->lock); genpd_unlock(genpd);
} }
return ret; return ret;
...@@ -936,13 +976,13 @@ static void pm_genpd_complete(struct device *dev) ...@@ -936,13 +976,13 @@ static void pm_genpd_complete(struct device *dev)
pm_generic_complete(dev); pm_generic_complete(dev);
mutex_lock(&genpd->lock); genpd_lock(genpd);
genpd->prepared_count--; genpd->prepared_count--;
if (!genpd->prepared_count) if (!genpd->prepared_count)
genpd_queue_power_off_work(genpd); genpd_queue_power_off_work(genpd);
mutex_unlock(&genpd->lock); genpd_unlock(genpd);
} }
/** /**
...@@ -1071,7 +1111,7 @@ static int genpd_add_device(struct generic_pm_domain *genpd, struct device *dev, ...@@ -1071,7 +1111,7 @@ static int genpd_add_device(struct generic_pm_domain *genpd, struct device *dev,
if (IS_ERR(gpd_data)) if (IS_ERR(gpd_data))
return PTR_ERR(gpd_data); return PTR_ERR(gpd_data);
mutex_lock(&genpd->lock); genpd_lock(genpd);
if (genpd->prepared_count > 0) { if (genpd->prepared_count > 0) {
ret = -EAGAIN; ret = -EAGAIN;
...@@ -1088,7 +1128,7 @@ static int genpd_add_device(struct generic_pm_domain *genpd, struct device *dev, ...@@ -1088,7 +1128,7 @@ static int genpd_add_device(struct generic_pm_domain *genpd, struct device *dev,
list_add_tail(&gpd_data->base.list_node, &genpd->dev_list); list_add_tail(&gpd_data->base.list_node, &genpd->dev_list);
out: out:
mutex_unlock(&genpd->lock); genpd_unlock(genpd);
if (ret) if (ret)
genpd_free_dev_data(dev, gpd_data); genpd_free_dev_data(dev, gpd_data);
...@@ -1130,7 +1170,7 @@ static int genpd_remove_device(struct generic_pm_domain *genpd, ...@@ -1130,7 +1170,7 @@ static int genpd_remove_device(struct generic_pm_domain *genpd,
gpd_data = to_gpd_data(pdd); gpd_data = to_gpd_data(pdd);
dev_pm_qos_remove_notifier(dev, &gpd_data->nb); dev_pm_qos_remove_notifier(dev, &gpd_data->nb);
mutex_lock(&genpd->lock); genpd_lock(genpd);
if (genpd->prepared_count > 0) { if (genpd->prepared_count > 0) {
ret = -EAGAIN; ret = -EAGAIN;
...@@ -1145,14 +1185,14 @@ static int genpd_remove_device(struct generic_pm_domain *genpd, ...@@ -1145,14 +1185,14 @@ static int genpd_remove_device(struct generic_pm_domain *genpd,
list_del_init(&pdd->list_node); list_del_init(&pdd->list_node);
mutex_unlock(&genpd->lock); genpd_unlock(genpd);
genpd_free_dev_data(dev, gpd_data); genpd_free_dev_data(dev, gpd_data);
return 0; return 0;
out: out:
mutex_unlock(&genpd->lock); genpd_unlock(genpd);
dev_pm_qos_add_notifier(dev, &gpd_data->nb); dev_pm_qos_add_notifier(dev, &gpd_data->nb);
return ret; return ret;
...@@ -1187,8 +1227,8 @@ static int genpd_add_subdomain(struct generic_pm_domain *genpd, ...@@ -1187,8 +1227,8 @@ static int genpd_add_subdomain(struct generic_pm_domain *genpd,
if (!link) if (!link)
return -ENOMEM; return -ENOMEM;
mutex_lock(&subdomain->lock); genpd_lock(subdomain);
mutex_lock_nested(&genpd->lock, SINGLE_DEPTH_NESTING); genpd_lock_nested(genpd, SINGLE_DEPTH_NESTING);
if (genpd->status == GPD_STATE_POWER_OFF if (genpd->status == GPD_STATE_POWER_OFF
&& subdomain->status != GPD_STATE_POWER_OFF) { && subdomain->status != GPD_STATE_POWER_OFF) {
...@@ -1211,8 +1251,8 @@ static int genpd_add_subdomain(struct generic_pm_domain *genpd, ...@@ -1211,8 +1251,8 @@ static int genpd_add_subdomain(struct generic_pm_domain *genpd,
genpd_sd_counter_inc(genpd); genpd_sd_counter_inc(genpd);
out: out:
mutex_unlock(&genpd->lock); genpd_unlock(genpd);
mutex_unlock(&subdomain->lock); genpd_unlock(subdomain);
if (ret) if (ret)
kfree(link); kfree(link);
return ret; return ret;
...@@ -1250,8 +1290,8 @@ int pm_genpd_remove_subdomain(struct generic_pm_domain *genpd, ...@@ -1250,8 +1290,8 @@ int pm_genpd_remove_subdomain(struct generic_pm_domain *genpd,
if (IS_ERR_OR_NULL(genpd) || IS_ERR_OR_NULL(subdomain)) if (IS_ERR_OR_NULL(genpd) || IS_ERR_OR_NULL(subdomain))
return -EINVAL; return -EINVAL;
mutex_lock(&subdomain->lock); genpd_lock(subdomain);
mutex_lock_nested(&genpd->lock, SINGLE_DEPTH_NESTING); genpd_lock_nested(genpd, SINGLE_DEPTH_NESTING);
if (!list_empty(&subdomain->master_links) || subdomain->device_count) { if (!list_empty(&subdomain->master_links) || subdomain->device_count) {
pr_warn("%s: unable to remove subdomain %s\n", genpd->name, pr_warn("%s: unable to remove subdomain %s\n", genpd->name,
...@@ -1275,8 +1315,8 @@ int pm_genpd_remove_subdomain(struct generic_pm_domain *genpd, ...@@ -1275,8 +1315,8 @@ int pm_genpd_remove_subdomain(struct generic_pm_domain *genpd,
} }
out: out:
mutex_unlock(&genpd->lock); genpd_unlock(genpd);
mutex_unlock(&subdomain->lock); genpd_unlock(subdomain);
return ret; return ret;
} }
...@@ -1316,7 +1356,8 @@ int pm_genpd_init(struct generic_pm_domain *genpd, ...@@ -1316,7 +1356,8 @@ int pm_genpd_init(struct generic_pm_domain *genpd,
INIT_LIST_HEAD(&genpd->master_links); INIT_LIST_HEAD(&genpd->master_links);
INIT_LIST_HEAD(&genpd->slave_links); INIT_LIST_HEAD(&genpd->slave_links);
INIT_LIST_HEAD(&genpd->dev_list); INIT_LIST_HEAD(&genpd->dev_list);
mutex_init(&genpd->lock); mutex_init(&genpd->mlock);
genpd->lock_ops = &genpd_mtx_ops;
genpd->gov = gov; genpd->gov = gov;
INIT_WORK(&genpd->power_off_work, genpd_power_off_work_fn); INIT_WORK(&genpd->power_off_work, genpd_power_off_work_fn);
atomic_set(&genpd->sd_count, 0); atomic_set(&genpd->sd_count, 0);
...@@ -1364,16 +1405,16 @@ static int genpd_remove(struct generic_pm_domain *genpd) ...@@ -1364,16 +1405,16 @@ static int genpd_remove(struct generic_pm_domain *genpd)
if (IS_ERR_OR_NULL(genpd)) if (IS_ERR_OR_NULL(genpd))
return -EINVAL; return -EINVAL;
mutex_lock(&genpd->lock); genpd_lock(genpd);
if (genpd->has_provider) { if (genpd->has_provider) {
mutex_unlock(&genpd->lock); genpd_unlock(genpd);
pr_err("Provider present, unable to remove %s\n", genpd->name); pr_err("Provider present, unable to remove %s\n", genpd->name);
return -EBUSY; return -EBUSY;
} }
if (!list_empty(&genpd->master_links) || genpd->device_count) { if (!list_empty(&genpd->master_links) || genpd->device_count) {
mutex_unlock(&genpd->lock); genpd_unlock(genpd);
pr_err("%s: unable to remove %s\n", __func__, genpd->name); pr_err("%s: unable to remove %s\n", __func__, genpd->name);
return -EBUSY; return -EBUSY;
} }
...@@ -1385,7 +1426,7 @@ static int genpd_remove(struct generic_pm_domain *genpd) ...@@ -1385,7 +1426,7 @@ static int genpd_remove(struct generic_pm_domain *genpd)
} }
list_del(&genpd->gpd_list_node); list_del(&genpd->gpd_list_node);
mutex_unlock(&genpd->lock); genpd_unlock(genpd);
cancel_work_sync(&genpd->power_off_work); cancel_work_sync(&genpd->power_off_work);
kfree(genpd->free); kfree(genpd->free);
pr_debug("%s: removed %s\n", __func__, genpd->name); pr_debug("%s: removed %s\n", __func__, genpd->name);
...@@ -1909,9 +1950,9 @@ int genpd_dev_pm_attach(struct device *dev) ...@@ -1909,9 +1950,9 @@ int genpd_dev_pm_attach(struct device *dev)
dev->pm_domain->detach = genpd_dev_pm_detach; dev->pm_domain->detach = genpd_dev_pm_detach;
dev->pm_domain->sync = genpd_dev_pm_sync; dev->pm_domain->sync = genpd_dev_pm_sync;
mutex_lock(&pd->lock); genpd_lock(pd);
ret = genpd_poweron(pd, 0); ret = genpd_poweron(pd, 0);
mutex_unlock(&pd->lock); genpd_unlock(pd);
out: out:
return ret ? -EPROBE_DEFER : 0; return ret ? -EPROBE_DEFER : 0;
} }
...@@ -2064,7 +2105,7 @@ static int pm_genpd_summary_one(struct seq_file *s, ...@@ -2064,7 +2105,7 @@ static int pm_genpd_summary_one(struct seq_file *s,
char state[16]; char state[16];
int ret; int ret;
ret = mutex_lock_interruptible(&genpd->lock); ret = genpd_lock_interruptible(genpd);
if (ret) if (ret)
return -ERESTARTSYS; return -ERESTARTSYS;
...@@ -2101,7 +2142,7 @@ static int pm_genpd_summary_one(struct seq_file *s, ...@@ -2101,7 +2142,7 @@ static int pm_genpd_summary_one(struct seq_file *s,
seq_puts(s, "\n"); seq_puts(s, "\n");
exit: exit:
mutex_unlock(&genpd->lock); genpd_unlock(genpd);
return 0; return 0;
} }
......
...@@ -42,13 +42,14 @@ struct genpd_power_state { ...@@ -42,13 +42,14 @@ struct genpd_power_state {
struct fwnode_handle *fwnode; struct fwnode_handle *fwnode;
}; };
struct genpd_lock_ops;
struct generic_pm_domain { struct generic_pm_domain {
struct dev_pm_domain domain; /* PM domain operations */ struct dev_pm_domain domain; /* PM domain operations */
struct list_head gpd_list_node; /* Node in the global PM domains list */ struct list_head gpd_list_node; /* Node in the global PM domains list */
struct list_head master_links; /* Links with PM domain as a master */ struct list_head master_links; /* Links with PM domain as a master */
struct list_head slave_links; /* Links with PM domain as a slave */ struct list_head slave_links; /* Links with PM domain as a slave */
struct list_head dev_list; /* List of devices */ struct list_head dev_list; /* List of devices */
struct mutex lock;
struct dev_power_governor *gov; struct dev_power_governor *gov;
struct work_struct power_off_work; struct work_struct power_off_work;
struct fwnode_handle *provider; /* Identity of the domain provider */ struct fwnode_handle *provider; /* Identity of the domain provider */
...@@ -74,6 +75,8 @@ struct generic_pm_domain { ...@@ -74,6 +75,8 @@ struct generic_pm_domain {
unsigned int state_count; /* number of states */ unsigned int state_count; /* number of states */
unsigned int state_idx; /* state that genpd will go to when off */ unsigned int state_idx; /* state that genpd will go to when off */
void *free; /* Free the state that was allocated for default */ void *free; /* Free the state that was allocated for default */
const struct genpd_lock_ops *lock_ops;
struct mutex mlock;
}; };
......
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