gpiolib: remove the GPIO device from the list when it's unregistered
authorBartosz Golaszewski <bartosz.golaszewski@linaro.org>
Tue, 2 Jan 2024 15:59:47 +0000 (16:59 +0100)
committerBartosz Golaszewski <bartosz.golaszewski@linaro.org>
Thu, 4 Jan 2024 09:29:16 +0000 (10:29 +0100)
If we wait until the GPIO device's .release() callback gets invoked
before we remove it from the global device list, then we risk that
someone will look it up using gpio_device_find() between where we
dropped the last reference and before .release() is done taking a
reference again to an object that's being released.

The device must be removed when it's being unregistered - just like how
we remove it from the GPIO bus.

Fixes: ff2b13592299 ("gpio: make the gpiochip a real device")
Signed-off-by: Bartosz Golaszewski <bartosz.golaszewski@linaro.org>
Reviewed-by: Linus Walleij <linus.walleij@linaro.org>
drivers/gpio/gpiolib.c

index e21497b989a19ef7f4fdf7c5a96e0f8394277ffa..e019c4243809bbcb11ac88dea131484026d7b94b 100644 (file)
@@ -651,9 +651,6 @@ static void gpiodev_release(struct device *dev)
 {
        struct gpio_device *gdev = to_gpio_device(dev);
 
-       scoped_guard(mutex, &gpio_devices_lock)
-               list_del(&gdev->list);
-
        ida_free(&gpio_ida, gdev->id);
        kfree_const(gdev->label);
        kfree(gdev->descs);
@@ -1068,6 +1065,9 @@ void gpiochip_remove(struct gpio_chip *gc)
                dev_crit(&gdev->dev,
                         "REMOVING GPIOCHIP WITH GPIOS STILL REQUESTED\n");
 
+       scoped_guard(mutex, &gpio_devices_lock)
+               list_del(&gdev->list);
+
        /*
         * The gpiochip side puts its use of the device to rest here:
         * if there are no userspace clients, the chardev and device will