Commit 54d9acd7 authored by Johan Hovold's avatar Johan Hovold Committed by Linus Walleij

gpio: sysfs: release irq after class-device deregistration

Make sure to release any irq only after the class device has been
deregistered.

This avoids a race between gpiod_unexport and edge_store, where an irq
could be allocated just before the gpio class device is deregistered
without relying on FLAG_EXPORT and the global sysfs lock.

Note that there is no need to hold the sysfs lock when releasing the irq
after the class device is gone as kernfs will prevent further attribute
operations.
Signed-off-by: default avatarJohan Hovold <johan@kernel.org>
Reviewed-by: default avatarAlexandre Courbot <acourbot@nvidia.com>
Signed-off-by: default avatarLinus Walleij <linus.walleij@linaro.org>
parent 6beac9d1
...@@ -695,7 +695,6 @@ void gpiod_unexport(struct gpio_desc *desc) ...@@ -695,7 +695,6 @@ void gpiod_unexport(struct gpio_desc *desc)
dev = class_find_device(&gpio_class, NULL, desc, match_export); dev = class_find_device(&gpio_class, NULL, desc, match_export);
if (dev) { if (dev) {
gpio_setup_irq(desc, dev, 0);
clear_bit(FLAG_SYSFS_DIR, &desc->flags); clear_bit(FLAG_SYSFS_DIR, &desc->flags);
clear_bit(FLAG_EXPORT, &desc->flags); clear_bit(FLAG_EXPORT, &desc->flags);
} else } else
...@@ -706,6 +705,11 @@ void gpiod_unexport(struct gpio_desc *desc) ...@@ -706,6 +705,11 @@ void gpiod_unexport(struct gpio_desc *desc)
if (dev) { if (dev) {
device_unregister(dev); device_unregister(dev);
/*
* Release irq after deregistration to prevent race with
* edge_store.
*/
gpio_setup_irq(desc, dev, 0);
put_device(dev); put_device(dev);
} }
......
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