Commit 459a25af authored by Boris BREZILLON's avatar Boris BREZILLON Committed by Thierry Reding

pwm: Get rid of pwm->lock

PWM devices are not protected against concurrent accesses. The lock in
struct pwm_device might let PWM users think it is, but it's actually
only protecting the enabled state.

Removing this lock should be fine as long as all PWM users are aware
that accesses to the PWM device have to be serialized, which seems to be
the case for all of them except the sysfs interface. Patch the sysfs
code by adding a lock to the pwm_export struct and making sure it's
taken for all relevant accesses to the exported PWM device.
Signed-off-by: default avatarBoris Brezillon <boris.brezillon@free-electrons.com>
Signed-off-by: default avatarThierry Reding <thierry.reding@gmail.com>
parent 4ff66efd
...@@ -269,7 +269,6 @@ int pwmchip_add_with_polarity(struct pwm_chip *chip, ...@@ -269,7 +269,6 @@ int pwmchip_add_with_polarity(struct pwm_chip *chip,
pwm->pwm = chip->base + i; pwm->pwm = chip->base + i;
pwm->hwpwm = i; pwm->hwpwm = i;
pwm->polarity = polarity; pwm->polarity = polarity;
mutex_init(&pwm->lock);
radix_tree_insert(&pwm_tree, pwm->pwm, pwm); radix_tree_insert(&pwm_tree, pwm->pwm, pwm);
} }
...@@ -474,22 +473,16 @@ int pwm_set_polarity(struct pwm_device *pwm, enum pwm_polarity polarity) ...@@ -474,22 +473,16 @@ int pwm_set_polarity(struct pwm_device *pwm, enum pwm_polarity polarity)
if (!pwm->chip->ops->set_polarity) if (!pwm->chip->ops->set_polarity)
return -ENOSYS; return -ENOSYS;
mutex_lock(&pwm->lock); if (pwm_is_enabled(pwm))
return -EBUSY;
if (pwm_is_enabled(pwm)) {
err = -EBUSY;
goto unlock;
}
err = pwm->chip->ops->set_polarity(pwm->chip, pwm, polarity); err = pwm->chip->ops->set_polarity(pwm->chip, pwm, polarity);
if (err) if (err)
goto unlock; return err;
pwm->polarity = polarity; pwm->polarity = polarity;
unlock: return 0;
mutex_unlock(&pwm->lock);
return err;
} }
EXPORT_SYMBOL_GPL(pwm_set_polarity); EXPORT_SYMBOL_GPL(pwm_set_polarity);
...@@ -506,16 +499,12 @@ int pwm_enable(struct pwm_device *pwm) ...@@ -506,16 +499,12 @@ int pwm_enable(struct pwm_device *pwm)
if (!pwm) if (!pwm)
return -EINVAL; return -EINVAL;
mutex_lock(&pwm->lock);
if (!test_and_set_bit(PWMF_ENABLED, &pwm->flags)) { if (!test_and_set_bit(PWMF_ENABLED, &pwm->flags)) {
err = pwm->chip->ops->enable(pwm->chip, pwm); err = pwm->chip->ops->enable(pwm->chip, pwm);
if (err) if (err)
clear_bit(PWMF_ENABLED, &pwm->flags); clear_bit(PWMF_ENABLED, &pwm->flags);
} }
mutex_unlock(&pwm->lock);
return err; return err;
} }
EXPORT_SYMBOL_GPL(pwm_enable); EXPORT_SYMBOL_GPL(pwm_enable);
......
...@@ -26,6 +26,7 @@ ...@@ -26,6 +26,7 @@
struct pwm_export { struct pwm_export {
struct device child; struct device child;
struct pwm_device *pwm; struct pwm_device *pwm;
struct mutex lock;
}; };
static struct pwm_export *child_to_pwm_export(struct device *child) static struct pwm_export *child_to_pwm_export(struct device *child)
...@@ -53,7 +54,8 @@ static ssize_t period_store(struct device *child, ...@@ -53,7 +54,8 @@ static ssize_t period_store(struct device *child,
struct device_attribute *attr, struct device_attribute *attr,
const char *buf, size_t size) const char *buf, size_t size)
{ {
struct pwm_device *pwm = child_to_pwm_device(child); struct pwm_export *export = child_to_pwm_export(child);
struct pwm_device *pwm = export->pwm;
unsigned int val; unsigned int val;
int ret; int ret;
...@@ -61,7 +63,9 @@ static ssize_t period_store(struct device *child, ...@@ -61,7 +63,9 @@ static ssize_t period_store(struct device *child,
if (ret) if (ret)
return ret; return ret;
mutex_lock(&export->lock);
ret = pwm_config(pwm, pwm_get_duty_cycle(pwm), val); ret = pwm_config(pwm, pwm_get_duty_cycle(pwm), val);
mutex_unlock(&export->lock);
return ret ? : size; return ret ? : size;
} }
...@@ -79,7 +83,8 @@ static ssize_t duty_cycle_store(struct device *child, ...@@ -79,7 +83,8 @@ static ssize_t duty_cycle_store(struct device *child,
struct device_attribute *attr, struct device_attribute *attr,
const char *buf, size_t size) const char *buf, size_t size)
{ {
struct pwm_device *pwm = child_to_pwm_device(child); struct pwm_export *export = child_to_pwm_export(child);
struct pwm_device *pwm = export->pwm;
unsigned int val; unsigned int val;
int ret; int ret;
...@@ -87,7 +92,9 @@ static ssize_t duty_cycle_store(struct device *child, ...@@ -87,7 +92,9 @@ static ssize_t duty_cycle_store(struct device *child,
if (ret) if (ret)
return ret; return ret;
mutex_lock(&export->lock);
ret = pwm_config(pwm, val, pwm_get_period(pwm)); ret = pwm_config(pwm, val, pwm_get_period(pwm));
mutex_unlock(&export->lock);
return ret ? : size; return ret ? : size;
} }
...@@ -105,13 +112,16 @@ static ssize_t enable_store(struct device *child, ...@@ -105,13 +112,16 @@ static ssize_t enable_store(struct device *child,
struct device_attribute *attr, struct device_attribute *attr,
const char *buf, size_t size) const char *buf, size_t size)
{ {
struct pwm_device *pwm = child_to_pwm_device(child); struct pwm_export *export = child_to_pwm_export(child);
struct pwm_device *pwm = export->pwm;
int val, ret; int val, ret;
ret = kstrtoint(buf, 0, &val); ret = kstrtoint(buf, 0, &val);
if (ret) if (ret)
return ret; return ret;
mutex_lock(&export->lock);
switch (val) { switch (val) {
case 0: case 0:
pwm_disable(pwm); pwm_disable(pwm);
...@@ -124,6 +134,8 @@ static ssize_t enable_store(struct device *child, ...@@ -124,6 +134,8 @@ static ssize_t enable_store(struct device *child,
break; break;
} }
mutex_unlock(&export->lock);
return ret ? : size; return ret ? : size;
} }
...@@ -151,7 +163,8 @@ static ssize_t polarity_store(struct device *child, ...@@ -151,7 +163,8 @@ static ssize_t polarity_store(struct device *child,
struct device_attribute *attr, struct device_attribute *attr,
const char *buf, size_t size) const char *buf, size_t size)
{ {
struct pwm_device *pwm = child_to_pwm_device(child); struct pwm_export *export = child_to_pwm_export(child);
struct pwm_device *pwm = export->pwm;
enum pwm_polarity polarity; enum pwm_polarity polarity;
int ret; int ret;
...@@ -162,7 +175,9 @@ static ssize_t polarity_store(struct device *child, ...@@ -162,7 +175,9 @@ static ssize_t polarity_store(struct device *child,
else else
return -EINVAL; return -EINVAL;
mutex_lock(&export->lock);
ret = pwm_set_polarity(pwm, polarity); ret = pwm_set_polarity(pwm, polarity);
mutex_unlock(&export->lock);
return ret ? : size; return ret ? : size;
} }
...@@ -203,6 +218,7 @@ static int pwm_export_child(struct device *parent, struct pwm_device *pwm) ...@@ -203,6 +218,7 @@ static int pwm_export_child(struct device *parent, struct pwm_device *pwm)
} }
export->pwm = pwm; export->pwm = pwm;
mutex_init(&export->lock);
export->child.release = pwm_export_release; export->child.release = pwm_export_release;
export->child.parent = parent; export->child.parent = parent;
......
...@@ -106,7 +106,6 @@ enum { ...@@ -106,7 +106,6 @@ enum {
* @pwm: global index of the PWM device * @pwm: global index of the PWM device
* @chip: PWM chip providing this PWM device * @chip: PWM chip providing this PWM device
* @chip_data: chip-private data associated with the PWM device * @chip_data: chip-private data associated with the PWM device
* @lock: used to serialize accesses to the PWM device where necessary
* @period: period of the PWM signal (in nanoseconds) * @period: period of the PWM signal (in nanoseconds)
* @duty_cycle: duty cycle of the PWM signal (in nanoseconds) * @duty_cycle: duty cycle of the PWM signal (in nanoseconds)
* @polarity: polarity of the PWM signal * @polarity: polarity of the PWM signal
...@@ -119,7 +118,6 @@ struct pwm_device { ...@@ -119,7 +118,6 @@ struct pwm_device {
unsigned int pwm; unsigned int pwm;
struct pwm_chip *chip; struct pwm_chip *chip;
void *chip_data; void *chip_data;
struct mutex lock;
unsigned int period; unsigned int period;
unsigned int duty_cycle; unsigned int duty_cycle;
......
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