Re: [PATCH] iio: adc: ad799x: use devm_iio_device_register and devm buffer setup

From: Jonathan Cameron

Date: Sun Mar 01 2026 - 06:37:24 EST


On Sat, 28 Feb 2026 21:15:15 +0530
Archit Anant <architanant5@xxxxxxxxx> wrote:

> Convert the driver to use the device-managed versions of
> iio_device_register() and iio_triggered_buffer_setup().
>
> This simplifies the error handling in ad799x_probe() by removing the
> 'error_cleanup_ring' goto label. It also removes the need to manually
> call iio_device_unregister() and iio_triggered_buffer_cleanup() in
> ad799x_remove().

It's also unfortunately broken. There is a simple rule of thumb for
use of devm. It must be used for 'everything' up to the point in probe()
where you decide to stop using it. After that it cannot be used
at all.

The reason is that everything in the remove() callback occurs before the
unwinding of devm actions. Thus if you mix and match, you get ordering
problems.

In this case you turn the power off before removing the user space
interfaces which is going to be rather unexpected if someone is still
using those interfaces.

Jonathan

>
> Signed-off-by: Archit Anant <architanant5@xxxxxxxxx>
> ---
> drivers/iio/adc/ad799x.c | 13 ++++---------
> 1 file changed, 4 insertions(+), 9 deletions(-)
>
> diff --git a/drivers/iio/adc/ad799x.c b/drivers/iio/adc/ad799x.c
> index 108bb22162ef..42712372acdb 100644
> --- a/drivers/iio/adc/ad799x.c
> +++ b/drivers/iio/adc/ad799x.c
> @@ -847,7 +847,7 @@ static int ad799x_probe(struct i2c_client *client)
> if (ret)
> goto error_disable_vref;
>
> - ret = iio_triggered_buffer_setup(indio_dev, NULL,
> + ret = devm_iio_triggered_buffer_setup(&client->dev, indio_dev, NULL,
> &ad799x_trigger_handler, NULL);
> if (ret)
> goto error_disable_vref;
> @@ -862,19 +862,17 @@ static int ad799x_probe(struct i2c_client *client)
> client->name,
> indio_dev);
> if (ret)
> - goto error_cleanup_ring;
> + goto error_disable_vref;
> }
>
> mutex_init(&st->lock);
>
> - ret = iio_device_register(indio_dev);
> + ret = devm_iio_device_register(&client->dev, indio_dev);
> if (ret)
> - goto error_cleanup_ring;
> + goto error_disable_vref;
>
> return 0;
>
> -error_cleanup_ring:
> - iio_triggered_buffer_cleanup(indio_dev);
> error_disable_vref:
> if (st->vref)
> regulator_disable(st->vref);
> @@ -889,9 +887,6 @@ static void ad799x_remove(struct i2c_client *client)
> struct iio_dev *indio_dev = i2c_get_clientdata(client);
> struct ad799x_state *st = iio_priv(indio_dev);
>
> - iio_device_unregister(indio_dev);
> -
> - iio_triggered_buffer_cleanup(indio_dev);
> if (st->vref)
> regulator_disable(st->vref);
> regulator_disable(st->reg);