On 23.02.2026 07:17, Tzung-Bi Shih wrote:
> Ensure struct gpio_chip for gpiochip_setup_dev().  This eliminates a few
> checks for struct gpio_chip.
>
> Signed-off-by: Tzung-Bi Shih <[email protected]>

This patch landed in today's linux-next as commit cf674f1a0c98 ("gpio: 
Ensure struct gpio_chip for gpiochip_setup_dev()"). In my tests I found 
that it triggers a warning on every test board I have, so I suspect that 
something is missing in the code. Here is an example of such warning:

------------[ cut here ]------------
WARNING: drivers/gpio/gpiolib-cdev.c:2735 at 
gpiolib_cdev_register+0x114/0x140, CPU#1: swapper/0/1
Modules linked in:
CPU: 1 UID: 0 PID: 1 Comm: swapper/0 Not tainted 
7.0.0-rc1-next-20260227-00065-g6af4b9cfeded #12259 PREEMPT
Hardware name: Samsung Exynos (Flattened Device Tree)
Call trace:
  unwind_backtrace from show_stack+0x10/0x14
  show_stack from dump_stack_lvl+0x68/0x88
  dump_stack_lvl from __warn+0x94/0x210
  __warn from warn_slowpath_fmt+0x1b0/0x1bc
  warn_slowpath_fmt from gpiolib_cdev_register+0x114/0x140
  gpiolib_cdev_register from gpiochip_setup_dev+0x4c/0xd0
  gpiochip_setup_dev from gpiochip_add_data_with_key+0x960/0xad4
  gpiochip_add_data_with_key from devm_gpiochip_add_data_with_key+0x20/0x5c
  devm_gpiochip_add_data_with_key from samsung_pinctrl_probe+0x8fc/0xbe8
  samsung_pinctrl_probe from platform_probe+0x5c/0x98
  platform_probe from really_probe+0xe0/0x424
  really_probe from __driver_probe_device+0x9c/0x1f4
  __driver_probe_device from driver_probe_device+0x30/0xc0
  driver_probe_device from __device_attach_driver+0xbc/0x180
  __device_attach_driver from bus_for_each_drv+0x84/0xdc
  bus_for_each_drv from __device_attach+0xb0/0x214
  __device_attach from device_initial_probe+0x3c/0x48
  device_initial_probe from bus_probe_device+0x24/0x80
  bus_probe_device from device_add+0x5c0/0x810
  device_add from of_platform_device_create_pdata+0xac/0x104
  of_platform_device_create_pdata from of_platform_bus_create+0x210/0x534
  of_platform_bus_create from of_platform_bus_create+0x27c/0x534
  of_platform_bus_create from of_platform_populate+0x90/0x150
  of_platform_populate from of_platform_default_populate_init+0xd0/0xe0
  of_platform_default_populate_init from do_one_initcall+0x70/0x3bc
  do_one_initcall from kernel_init_freeable+0x1c0/0x248
  kernel_init_freeable from kernel_init+0x18/0x12c
  kernel_init from ret_from_fork+0x14/0x28
Exception stack(0xf082dfb0 to 0xf082dff8)
dfa0:                                     00000000 00000000 00000000 
00000000
dfc0: 00000000 00000000 00000000 00000000 00000000 00000000 00000000 
00000000
dfe0: 00000000 00000000 00000000 00000000 00000013 00000000
irq event stamp: 55167
hardirqs last  enabled at (55387): [<c01c3578>] __up_console_sem+0x50/0x60
hardirqs last disabled at (55398): [<c01c3564>] __up_console_sem+0x3c/0x60
softirqs last  enabled at (55352): [<c013c8fc>] handle_softirqs+0x32c/0x5c0
softirqs last disabled at (55419): [<c013cd3c>] __irq_exit_rcu+0x144/0x1f0
---[ end trace 0000000000000000 ]---


> ---
> v4:
> - To be consistent, rename `chip` -> `gc`.
> - Add lockdep checks.
>
> v3: https://lore.kernel.org/all/[email protected]
> - Pass struct gpio_chip * only.
>
> v2: https://lore.kernel.org/all/[email protected]
> - No changes.
>
> v1: https://lore.kernel.org/all/[email protected]
>
>   drivers/gpio/gpiolib-cdev.c  | 14 ++++----------
>   drivers/gpio/gpiolib-cdev.h  |  2 +-
>   drivers/gpio/gpiolib-sysfs.c | 21 ++++++++-------------
>   drivers/gpio/gpiolib-sysfs.h |  4 ++--
>   drivers/gpio/gpiolib.c       | 24 +++++++++++++++++-------
>   5 files changed, 32 insertions(+), 33 deletions(-)
>
> diff --git a/drivers/gpio/gpiolib-cdev.c b/drivers/gpio/gpiolib-cdev.c
> index 73ae77f0f213..a154b04e9316 100644
> --- a/drivers/gpio/gpiolib-cdev.c
> +++ b/drivers/gpio/gpiolib-cdev.c
> @@ -2782,11 +2782,13 @@ static const struct file_operations gpio_fileops = {
>   #endif
>   };
>   
> -int gpiolib_cdev_register(struct gpio_device *gdev, dev_t devt)
> +int gpiolib_cdev_register(struct gpio_chip *gc, dev_t devt)
>   {
> -     struct gpio_chip *gc;
> +     struct gpio_device *gdev = gc->gpiodev;
>       int ret;
>   
> +     lockdep_assert_held(&gdev->srcu);
> +
>       cdev_init(&gdev->chrdev, &gpio_fileops);
>       gdev->chrdev.owner = THIS_MODULE;
>       gdev->dev.devt = MKDEV(MAJOR(devt), gdev->id);
> @@ -2802,14 +2804,6 @@ int gpiolib_cdev_register(struct gpio_device *gdev, 
> dev_t devt)
>               return ret;
>       }
>   
> -     guard(srcu)(&gdev->srcu);
> -     gc = srcu_dereference(gdev->chip, &gdev->srcu);
> -     if (!gc) {
> -             cdev_device_del(&gdev->chrdev, &gdev->dev);
> -             destroy_workqueue(gdev->line_state_wq);
> -             return -ENODEV;
> -     }
> -
>       gpiochip_dbg(gc, "added GPIO chardev (%d:%d)\n", MAJOR(devt), gdev->id);
>   
>       return 0;
> diff --git a/drivers/gpio/gpiolib-cdev.h b/drivers/gpio/gpiolib-cdev.h
> index b42644cbffb8..4a9cb3335d99 100644
> --- a/drivers/gpio/gpiolib-cdev.h
> +++ b/drivers/gpio/gpiolib-cdev.h
> @@ -7,7 +7,7 @@
>   
>   struct gpio_device;
>   
> -int gpiolib_cdev_register(struct gpio_device *gdev, dev_t devt);
> +int gpiolib_cdev_register(struct gpio_chip *gc, dev_t devt);
>   void gpiolib_cdev_unregister(struct gpio_device *gdev);
>   
>   #endif /* GPIOLIB_CDEV_H */
> diff --git a/drivers/gpio/gpiolib-sysfs.c b/drivers/gpio/gpiolib-sysfs.c
> index 1c25a7dd3db4..748a3eb1bf35 100644
> --- a/drivers/gpio/gpiolib-sysfs.c
> +++ b/drivers/gpio/gpiolib-sysfs.c
> @@ -983,13 +983,15 @@ void gpiod_unexport(struct gpio_desc *desc)
>   }
>   EXPORT_SYMBOL_GPL(gpiod_unexport);
>   
> -int gpiochip_sysfs_register(struct gpio_device *gdev)
> +int gpiochip_sysfs_register(struct gpio_chip *gc)
>   {
> +     struct gpio_device *gdev = gc->gpiodev;
>       struct gpiodev_data *data;
> -     struct gpio_chip *chip;
>       struct device *parent;
>       int err;
>   
> +     lockdep_assert_held(&gdev->srcu);
> +
>       /*
>        * Many systems add gpio chips for SOC support very early,
>        * before driver model support is available.  In those cases we
> @@ -999,18 +1001,12 @@ int gpiochip_sysfs_register(struct gpio_device *gdev)
>       if (!class_is_registered(&gpio_class))
>               return 0;
>   
> -     guard(srcu)(&gdev->srcu);
> -
> -     chip = srcu_dereference(gdev->chip, &gdev->srcu);
> -     if (!chip)
> -             return -ENODEV;
> -
>       /*
>        * For sysfs backward compatibility we need to preserve this
>        * preferred parenting to the gpio_chip parent field, if set.
>        */
> -     if (chip->parent)
> -             parent = chip->parent;
> +     if (gc->parent)
> +             parent = gc->parent;
>       else
>               parent = &gdev->dev;
>   
> @@ -1029,7 +1025,7 @@ int gpiochip_sysfs_register(struct gpio_device *gdev)
>                                                   MKDEV(0, 0), data,
>                                                   gpiochip_groups,
>                                                   GPIOCHIP_NAME "%d",
> -                                                 chip->base);
> +                                                 gc->base);
>       if (IS_ERR(data->cdev_base)) {
>               err = PTR_ERR(data->cdev_base);
>               kfree(data);
> @@ -1085,10 +1081,9 @@ void gpiochip_sysfs_unregister(struct gpio_chip *gc)
>    */
>   static int gpiofind_sysfs_register(struct gpio_chip *gc, const void *data)
>   {
> -     struct gpio_device *gdev = gc->gpiodev;
>       int ret;
>   
> -     ret = gpiochip_sysfs_register(gdev);
> +     ret = gpiochip_sysfs_register(gc);
>       if (ret)
>               gpiochip_err(gc, "failed to register the sysfs entry: %d\n", 
> ret);
>   
> diff --git a/drivers/gpio/gpiolib-sysfs.h b/drivers/gpio/gpiolib-sysfs.h
> index fd5db5384681..d0998de043a2 100644
> --- a/drivers/gpio/gpiolib-sysfs.h
> +++ b/drivers/gpio/gpiolib-sysfs.h
> @@ -7,12 +7,12 @@ struct gpio_device;
>   
>   #ifdef CONFIG_GPIO_SYSFS
>   
> -int gpiochip_sysfs_register(struct gpio_device *gdev);
> +int gpiochip_sysfs_register(struct gpio_chip *gc);
>   void gpiochip_sysfs_unregister(struct gpio_chip *gc);
>   
>   #else
>   
> -static inline int gpiochip_sysfs_register(struct gpio_device *gdev)
> +static inline int gpiochip_sysfs_register(struct gpio_chip *gc)
>   {
>       return 0;
>   }
> diff --git a/drivers/gpio/gpiolib.c b/drivers/gpio/gpiolib.c
> index d5070c538ba5..44635e9a29c3 100644
> --- a/drivers/gpio/gpiolib.c
> +++ b/drivers/gpio/gpiolib.c
> @@ -881,14 +881,14 @@ static const struct device_type gpio_dev_type = {
>   };
>   
>   #ifdef CONFIG_GPIO_CDEV
> -#define gcdev_register(gdev, devt)   gpiolib_cdev_register((gdev), (devt))
> +#define gcdev_register(gc, devt)     gpiolib_cdev_register((gc), (devt))
>   #define gcdev_unregister(gdev)              gpiolib_cdev_unregister((gdev))
>   #else
>   /*
>    * gpiolib_cdev_register() indirectly calls device_add(), which is still
>    * required even when cdev is not selected.
>    */
> -#define gcdev_register(gdev, devt)   device_add(&(gdev)->dev)
> +#define gcdev_register(gc, devt)     device_add(&(gc)->gpiodev->dev)
>   #define gcdev_unregister(gdev)              device_del(&(gdev)->dev)
>   #endif
>   
> @@ -896,8 +896,9 @@ static const struct device_type gpio_dev_type = {
>    * An initial reference count has been held in gpiochip_add_data_with_key().
>    * The caller should drop the reference via gpio_device_put() on errors.
>    */
> -static int gpiochip_setup_dev(struct gpio_device *gdev)
> +static int gpiochip_setup_dev(struct gpio_chip *gc)
>   {
> +     struct gpio_device *gdev = gc->gpiodev;
>       struct fwnode_handle *fwnode = dev_fwnode(&gdev->dev);
>       int ret;
>   
> @@ -910,11 +911,11 @@ static int gpiochip_setup_dev(struct gpio_device *gdev)
>       if (fwnode && !fwnode->dev)
>               fwnode_dev_initialized(fwnode, false);
>   
> -     ret = gcdev_register(gdev, gpio_devt);
> +     ret = gcdev_register(gc, gpio_devt);
>       if (ret)
>               return ret;
>   
> -     ret = gpiochip_sysfs_register(gdev);
> +     ret = gpiochip_sysfs_register(gc);
>       if (ret)
>               goto err_remove_device;
>   
> @@ -961,13 +962,22 @@ static void machine_gpiochip_add(struct gpio_chip *gc)
>   static void gpiochip_setup_devs(void)
>   {
>       struct gpio_device *gdev;
> +     struct gpio_chip *gc;
>       int ret;
>   
>       guard(srcu)(&gpio_devices_srcu);
>   
>       list_for_each_entry_srcu(gdev, &gpio_devices, list,
>                                srcu_read_lock_held(&gpio_devices_srcu)) {
> -             ret = gpiochip_setup_dev(gdev);
> +             guard(srcu)(&gdev->srcu);
> +
> +             gc = srcu_dereference(gdev->chip, &gdev->srcu);
> +             if (!gc) {
> +                     dev_err(&gdev->dev, "Underlying GPIO chip is gone\n");
> +                     continue;
> +             }
> +
> +             ret = gpiochip_setup_dev(gc);
>               if (ret) {
>                       gpio_device_put(gdev);
>                       dev_err(&gdev->dev,
> @@ -1225,7 +1235,7 @@ int gpiochip_add_data_with_key(struct gpio_chip *gc, 
> void *data,
>        * (i.e., `gpio_bus_type` is ready).  Otherwise, defer until later.
>        */
>       if (gpiolib_initialized) {
> -             ret = gpiochip_setup_dev(gdev);
> +             ret = gpiochip_setup_dev(gc);
>               if (ret)
>                       goto err_teardown_shared;
>       }

Best regards
-- 
Marek Szyprowski, PhD
Samsung R&D Institute Poland


Reply via email to