diff mbox series

[v2,09/10] iio: adc: ad7606: Add iio-backend support

Message ID 20240920-ad7606_add_iio_backend_support-v2-9-0e78782ae7d0@baylibre.com (mailing list archive)
State Changes Requested
Headers show
Series Add iio backend compatibility for ad7606 | expand

Commit Message

Guillaume Stols Sept. 20, 2024, 5:33 p.m. UTC
- Basic support for iio backend.
- Supports IIO_CHAN_INFO_SAMP_FREQ R/W.
- Only hardware mode is available, and that IIO_CHAN_INFO_RAW is not
  supported if iio-backend mode is selected.

Signed-off-by: Guillaume Stols <gstols@baylibre.com>
---
 drivers/iio/adc/Kconfig      |  2 +
 drivers/iio/adc/ad7606.c     | 94 +++++++++++++++++++++++++++++++++++++-------
 drivers/iio/adc/ad7606.h     | 15 +++++++
 drivers/iio/adc/ad7606_par.c | 91 ++++++++++++++++++++++++++++++++++++++++++
 4 files changed, 187 insertions(+), 15 deletions(-)

Comments

David Lechner Sept. 23, 2024, 9:40 a.m. UTC | #1
On Fri, Sep 20, 2024 at 7:33 PM Guillaume Stols <gstols@baylibre.com> wrote:
>
> - Basic support for iio backend.
> - Supports IIO_CHAN_INFO_SAMP_FREQ R/W.
> - Only hardware mode is available, and that IIO_CHAN_INFO_RAW is not
>   supported if iio-backend mode is selected.
>
> Signed-off-by: Guillaume Stols <gstols@baylibre.com>
> ---
>  drivers/iio/adc/Kconfig      |  2 +
>  drivers/iio/adc/ad7606.c     | 94 +++++++++++++++++++++++++++++++++++++-------
>  drivers/iio/adc/ad7606.h     | 15 +++++++
>  drivers/iio/adc/ad7606_par.c | 91 ++++++++++++++++++++++++++++++++++++++++++
>  4 files changed, 187 insertions(+), 15 deletions(-)
>
> diff --git a/drivers/iio/adc/Kconfig b/drivers/iio/adc/Kconfig
> index 4ab1a3092d88..9b52d5b2c592 100644
> --- a/drivers/iio/adc/Kconfig
> +++ b/drivers/iio/adc/Kconfig
> @@ -224,9 +224,11 @@ config AD7606_IFACE_PARALLEL
>         tristate "Analog Devices AD7606 ADC driver with parallel interface support"
>         depends on HAS_IOPORT
>         select AD7606
> +       select IIO_BACKEND
>         help
>           Say yes here to build parallel interface support for Analog Devices:
>           ad7605-4, ad7606, ad7606-6, ad7606-4 analog to digital converters (ADC).
> +         It also support iio_backended devices for AD7606B.
>
>           To compile this driver as a module, choose M here: the
>           module will be called ad7606_par.
> diff --git a/drivers/iio/adc/ad7606.c b/drivers/iio/adc/ad7606.c
> index 7f2ff1674638..f710445bdc22 100644
> --- a/drivers/iio/adc/ad7606.c
> +++ b/drivers/iio/adc/ad7606.c
> @@ -21,6 +21,7 @@
>  #include <linux/util_macros.h>
>  #include <linux/units.h>
>
> +#include <linux/iio/backend.h>
>  #include <linux/iio/iio.h>
>  #include <linux/iio/buffer.h>
>  #include <linux/iio/sysfs.h>
> @@ -271,7 +272,15 @@ static int ad7606_set_sampling_freq(struct ad7606_state *st, unsigned long freq)
>
>  static int ad7606_read_samples(struct ad7606_state *st)
>  {
> -       unsigned int num = st->chip_info->num_channels - 1;
> +       unsigned int num = st->chip_info->num_channels;

Probably better to introduce a new num_voltage_channels field to
chip_info instead of trying to reverse engineer if there is a
timestamp channel or not.

> +
> +       /*
> +        * Timestamp channel does not contain sample, and no timestamp channel if
> +        * backend is used.
> +        */
> +       if (!st->back)
> +               num--;
> +
>         u16 *data = st->data;
>
>         return st->bops->read_block(st->dev, num, data);
> @@ -319,11 +328,14 @@ static int ad7606_scan_direct(struct iio_dev *indio_dev, unsigned int ch)
>                 if (!ret)
>                         return ret;
>         }
> -       ret = wait_for_completion_timeout(&st->completion,
> -                                         msecs_to_jiffies(1000));
> -       if (!ret) {
> -               ret = -ETIMEDOUT;
> -               goto error_ret;
> +
> +       if (!st->back) {
> +               ret = wait_for_completion_timeout(&st->completion,
> +                                                 msecs_to_jiffies(1000));
> +               if (!ret) {
> +                       ret = -ETIMEDOUT;
> +                       goto error_ret;
> +               }
>         }

Would it be better to just make a different scan_direct function for
the case when we don't have the BUSY interrtup?

>
>         ret = ad7606_read_samples(st);
> @@ -349,6 +361,7 @@ static int ad7606_read_raw(struct iio_dev *indio_dev,
>  {
>         int ret, ch = 0;
>         struct ad7606_state *st = iio_priv(indio_dev);
> +       struct pwm_state cnvst_pwm_state;
>
>         switch (m) {
>         case IIO_CHAN_INFO_RAW:
> @@ -369,6 +382,10 @@ static int ad7606_read_raw(struct iio_dev *indio_dev,
>         case IIO_CHAN_INFO_OVERSAMPLING_RATIO:
>                 *val = st->oversampling;
>                 return IIO_VAL_INT;
> +       case IIO_CHAN_INFO_SAMP_FREQ:
> +               pwm_get_state(st->cnvst_pwm, &cnvst_pwm_state);
> +               *val = DIV_ROUND_CLOSEST_ULL(NSEC_PER_SEC, cnvst_pwm_state.period);
> +               return IIO_VAL_INT;
>         }
>         return -EINVAL;
>  }
> @@ -458,6 +475,8 @@ static int ad7606_write_raw(struct iio_dev *indio_dev,
>                         return ret;
>
>                 return 0;
> +       case IIO_CHAN_INFO_SAMP_FREQ:
> +               return ad7606_set_sampling_freq(st, val);
>         default:
>                 return -EINVAL;
>         }
> @@ -595,14 +614,49 @@ static int ad7606_buffer_predisable(struct iio_dev *indio_dev)
>         return 0;
>  }
>
> +static int ad7606_pwm_buffer_postenable(struct iio_dev *indio_dev)
> +{
> +       struct ad7606_state *st = iio_priv(indio_dev);
> +
> +       return ad7606_pwm_set_swing(st);
> +}
> +
> +static int ad7606_pwm_buffer_predisable(struct iio_dev *indio_dev)
> +{
> +       struct ad7606_state *st = iio_priv(indio_dev);
> +
> +       return ad7606_pwm_set_low(st);
> +}
> +
> +static int ad7606_update_scan_mode(struct iio_dev *indio_dev,
> +                                  const unsigned long *scan_mask)
> +{
> +       struct ad7606_state *st = iio_priv(indio_dev);
> +
> +       /* The update scan mode is only for iio backend compatible drivers.
> +        * If the specific update_scan_mode is not defined in the bus ops,
> +        * just do nothing and return 0.
> +        */
> +       if (st->bops->update_scan_mode)
> +               return st->bops->update_scan_mode(indio_dev, scan_mask);
> +       else
> +               return 0;
> +}

Usually, we try to avoid else after return.

So perhaps simpler:

if (!st->bops->update_scan_mode)
        return 0;

return st->bops->update_scan_mode(indio_dev, scan_mask);

> +
>  static const struct iio_buffer_setup_ops ad7606_buffer_ops = {
>         .postenable = &ad7606_buffer_postenable,
>         .predisable = &ad7606_buffer_predisable,
>  };
>
> +static const struct iio_buffer_setup_ops ad7606_pwm_buffer_ops = {
> +       .postenable = &ad7606_pwm_buffer_postenable,
> +       .predisable = &ad7606_pwm_buffer_predisable,
> +};
> +
>  static const struct iio_info ad7606_info_no_os_or_range = {
>         .read_raw = &ad7606_read_raw,
>         .validate_trigger = &ad7606_validate_trigger,
> +       .update_scan_mode = &ad7606_update_scan_mode,
>  };
>
>  static const struct iio_info ad7606_info_os_and_range = {
> @@ -610,6 +664,7 @@ static const struct iio_info ad7606_info_os_and_range = {
>         .write_raw = &ad7606_write_raw,
>         .attrs = &ad7606_attribute_group_os_and_range,
>         .validate_trigger = &ad7606_validate_trigger,
> +       .update_scan_mode = &ad7606_update_scan_mode,
>  };
>
>  static const struct iio_info ad7606_info_os_range_and_debug = {
> @@ -618,6 +673,7 @@ static const struct iio_info ad7606_info_os_range_and_debug = {
>         .debugfs_reg_access = &ad7606_reg_access,
>         .attrs = &ad7606_attribute_group_os_and_range,
>         .validate_trigger = &ad7606_validate_trigger,
> +       .update_scan_mode = &ad7606_update_scan_mode,
>  };
>
>  static const struct iio_info ad7606_info_os = {
> @@ -625,6 +681,7 @@ static const struct iio_info ad7606_info_os = {
>         .write_raw = &ad7606_write_raw,
>         .attrs = &ad7606_attribute_group_os,
>         .validate_trigger = &ad7606_validate_trigger,
> +       .update_scan_mode = &ad7606_update_scan_mode,
>  };
>
>  static const struct iio_info ad7606_info_range = {
> @@ -632,6 +689,7 @@ static const struct iio_info ad7606_info_range = {
>         .write_raw = &ad7606_write_raw,
>         .attrs = &ad7606_attribute_group_range,
>         .validate_trigger = &ad7606_validate_trigger,
> +       .update_scan_mode = &ad7606_update_scan_mode,
>  };
>
>  static const struct iio_trigger_ops ad7606_trigger_ops = {
> @@ -700,8 +758,6 @@ int ad7606_probe(struct device *dev, int irq, void __iomem *base_address,
>         indio_dev->channels = st->chip_info->channels;
>         indio_dev->num_channels = st->chip_info->num_channels;
>
> -       init_completion(&st->completion);
> -
>         ret = ad7606_reset(st);
>         if (ret)
>                 dev_warn(st->dev, "failed to RESET: no RESET GPIO specified\n");
> @@ -774,14 +830,22 @@ int ad7606_probe(struct device *dev, int irq, void __iomem *base_address,
>                 if (ret)
>                         return ret;
>         }
> -       ret = devm_request_threaded_irq(dev, irq,
> -                                       NULL,
> -                                       &ad7606_interrupt,
> -                                       IRQF_TRIGGER_FALLING | IRQF_ONESHOT,
> -                                       chip_info->name, indio_dev);
> -       if (ret)
> -               return ret;
>
> +       if (st->bops->iio_backend_config) {
> +               ret = st->bops->iio_backend_config(dev, indio_dev);
> +               if (ret)
> +                       return ret;
> +               indio_dev->setup_ops = &ad7606_pwm_buffer_ops;
> +       } else {
> +               init_completion(&st->completion);
> +               ret = devm_request_threaded_irq(dev, irq,
> +                                               NULL,
> +                                               &ad7606_interrupt,
> +                                               IRQF_TRIGGER_FALLING | IRQF_ONESHOT,
> +                                               chip_info->name, indio_dev);
> +               if (ret)
> +                       return ret;
> +       }
>         return devm_iio_device_register(dev, indio_dev);
>  }
>  EXPORT_SYMBOL_NS_GPL(ad7606_probe, IIO_AD7606);
> diff --git a/drivers/iio/adc/ad7606.h b/drivers/iio/adc/ad7606.h
> index 18c87fe9a41a..53cd8eb4898e 100644
> --- a/drivers/iio/adc/ad7606.h
> +++ b/drivers/iio/adc/ad7606.h
> @@ -34,6 +34,12 @@
>                 BIT(IIO_CHAN_INFO_SCALE),               \
>                 BIT(IIO_CHAN_INFO_OVERSAMPLING_RATIO))
>
> +#define AD7606_BI_CHANNEL(num)                         \
> +       AD760X_CHANNEL(num, 0,                          \
> +               BIT(IIO_CHAN_INFO_SCALE),               \
> +               BIT(IIO_CHAN_INFO_SAMP_FREQ) |          \
> +               BIT(IIO_CHAN_INFO_OVERSAMPLING_RATIO))
> +
>  #define AD7616_CHANNEL(num)    \
>         AD760X_CHANNEL(num, BIT(IIO_CHAN_INFO_RAW) | BIT(IIO_CHAN_INFO_SCALE),\
>                 0, BIT(IIO_CHAN_INFO_OVERSAMPLING_RATIO))
> @@ -59,6 +65,7 @@ enum ad7606_supported_device_ids {
>   * @os_req_reset       some devices require a reset to update oversampling
>   * @init_delay_ms      required delay in miliseconds for initialization
>   *                     after a restart
> + * @has_backend                defines if a backend is available for the given chip
>   */
>  struct ad7606_chip_info {
>         enum ad7606_supported_device_ids id;
> @@ -69,6 +76,7 @@ struct ad7606_chip_info {
>         unsigned int                    oversampling_num;
>         bool                            os_req_reset;
>         unsigned long                   init_delay_ms;
> +       bool                            has_backend;

It isn't clear to me what this flag is for. If the flag is true, does
it mean that the chip requires to use the IIO backend and not the
older parallel interface? What if there is some chip that need to
support both?

>  };
>
>  /**
> @@ -115,6 +123,7 @@ struct ad7606_state {
>         unsigned int                    num_scales;
>         const unsigned int              *oversampling_avail;
>         unsigned int                    num_os_ratios;
> +       struct iio_backend              *back;
>         int (*write_scale)(struct iio_dev *indio_dev, int ch, int val);
>         int (*write_os)(struct iio_dev *indio_dev, int val);
>
> @@ -139,16 +148,21 @@ struct ad7606_state {
>
>  /**
>   * struct ad7606_bus_ops - driver bus operations
> + * @iio_backend_config function pointer for configuring the iio_backend for
> + *                     the compatibles that use it
>   * @read_block         function pointer for reading blocks of data
>   * @sw_mode_config:    pointer to a function which configured the device
>   *                     for software mode
>   * @reg_read   function pointer for reading spi register
>   * @reg_write  function pointer for writing spi register
>   * @write_mask function pointer for write spi register with mask
> + * @update_scan_mode   function pointer for handling the calls to iio_info's update_scan
> + *                     mode when enabling/disabling channels.
>   * @rd_wr_cmd  pointer to the function which calculates the spi address
>   */
>  struct ad7606_bus_ops {
>         /* more methods added in future? */
> +       int (*iio_backend_config)(struct device *dev, struct iio_dev *indio_dev);
>         int (*read_block)(struct device *dev, int num, void *data);
>         int (*sw_mode_config)(struct iio_dev *indio_dev);
>         int (*reg_read)(struct ad7606_state *st, unsigned int addr);
> @@ -159,6 +173,7 @@ struct ad7606_bus_ops {
>                                  unsigned int addr,
>                                  unsigned long mask,
>                                  unsigned int val);
> +       int (*update_scan_mode)(struct iio_dev *indio_dev, const unsigned long *scan_mask);
>         u16 (*rd_wr_cmd)(int addr, char isWriteOp);
>  };
>
> diff --git a/drivers/iio/adc/ad7606_par.c b/drivers/iio/adc/ad7606_par.c
> index 7bac39033955..564284ede997 100644
> --- a/drivers/iio/adc/ad7606_par.c
> +++ b/drivers/iio/adc/ad7606_par.c
> @@ -3,6 +3,8 @@
>   * AD7606 Parallel Interface ADC driver
>   *
>   * Copyright 2011 Analog Devices Inc.
> + * Copyright 2024 Analog Devices Inc.

Can just add year to existing copyright line.

> + * Copyright 2024 BayLibre SAS.
>   */
>
>  #include <linux/err.h>
> @@ -15,8 +17,80 @@
>  #include <linux/types.h>
>
>  #include <linux/iio/iio.h>
> +#include <linux/iio/backend.h>

Alphabetical order?

> +
>  #include "ad7606.h"
>
> +static const struct iio_chan_spec ad7606b_bi_channels[] = {
> +       AD7606_BI_CHANNEL(0),
> +       AD7606_BI_CHANNEL(1),
> +       AD7606_BI_CHANNEL(2),
> +       AD7606_BI_CHANNEL(3),
> +       AD7606_BI_CHANNEL(4),
> +       AD7606_BI_CHANNEL(5),
> +       AD7606_BI_CHANNEL(6),
> +       AD7606_BI_CHANNEL(7),
> +};
> +
> +static int ad7606_bi_update_scan_mode(struct iio_dev *indio_dev, const unsigned long *scan_mask)
> +{
> +       struct ad7606_state *st = iio_priv(indio_dev);
> +       unsigned int c, ret;
> +
> +       for (c = 0; c < indio_dev->num_channels; c++) {
> +               if (test_bit(c, scan_mask))
> +                       ret = iio_backend_chan_enable(st->back, c);
> +               else
> +                       ret = iio_backend_chan_disable(st->back, c);
> +               if (ret)
> +                       return ret;
> +       }
> +
> +       return 0;
> +}
> +
> +static int ad7606_bi_setup_iio_backend(struct device *dev, struct iio_dev *indio_dev)
> +{
> +       struct ad7606_state *st = iio_priv(indio_dev);
> +       unsigned int ret, c;
> +       struct iio_backend_data_fmt data = {
> +               .sign_extend = true,
> +               .enable = true,
> +       };
> +
> +       st->back = devm_iio_backend_get(dev, NULL);
> +       if (IS_ERR(st->back))
> +               return PTR_ERR(st->back);
> +
> +       /* If the device is iio_backend powered the PWM is mandatory */
> +       if (!st->cnvst_pwm)
> +               return -EINVAL;

Probably useful to print an error message here since EINVAL can be a
lot of things.

> +
> +       ret = devm_iio_backend_request_buffer(dev, st->back, indio_dev);
> +       if (ret)
> +               return ret;
> +
> +       ret = devm_iio_backend_enable(dev, st->back);
> +       if (ret)
> +               return ret;
> +
> +       for (c = 0; c < indio_dev->num_channels; c++) {
> +               ret = iio_backend_data_format_set(st->back, c, &data);
> +               if (ret)
> +                       return ret;
> +       }
> +
> +       indio_dev->channels = ad7606b_bi_channels;
> +       indio_dev->num_channels = 8;

Can use ARRAY_SIZE(ad7606b_bi_channels) instead of hard-coding 8.

> +
> +       return 0;
> +}
> +
> +static const struct ad7606_bus_ops ad7606_bi_bops = {
> +       .iio_backend_config = ad7606_bi_setup_iio_backend,
> +       .update_scan_mode = ad7606_bi_update_scan_mode,
> +};
> +
>  static int ad7606_par16_read_block(struct device *dev,
>                                    int count, void *buf)
>  {
> @@ -96,9 +170,23 @@ static int ad7606_par_probe(struct platform_device *pdev)
>         void __iomem *addr;
>         resource_size_t remap_size;
>         int irq;
> +       struct iio_backend *back;
>
> +       /*
> +        * If a firmware node is available (ACPI or DT), platform_device_id is null
> +        * and we must use get_match_data.
> +        */
>         if (dev_fwnode(&pdev->dev)) {
>                 chip_info = device_get_match_data(&pdev->dev);
> +               back = devm_iio_backend_get(&pdev->dev, NULL);
> +               if (!IS_ERR(back))
> +                       /*
> +                        * If a backend is available in the device tree, call the core
> +                        * probe with backend bops, otherwise use the former bops.
> +                        */
> +                       return ad7606_probe(&pdev->dev, 0, NULL,
> +                                           chip_info,
> +                                           &ad7606_bi_bops);

It seems strange to be this adding inside the if statement for the DT
case. It would be more future proof to have it after instead, e.g. if
you bring back the patch for ad7606b_bi_channels().

>         } else {
>                 id = platform_get_device_id(pdev);
>                 chip_info = (const struct ad7606_chip_info *)id->driver_data;
> @@ -125,6 +213,7 @@ static const struct platform_device_id ad7606_driver_ids[] = {
>         { .name = "ad7606-4", .driver_data = (kernel_ulong_t)&ad7606_4_info, },
>         { .name = "ad7606-6", .driver_data = (kernel_ulong_t)&ad7606_6_info, },
>         { .name = "ad7606-8", .driver_data = (kernel_ulong_t)&ad7606_8_info, },
> +       { .name = "ad7606b", .driver_data = (kernel_ulong_t)&ad7606b_info, },
>         { }
>  };
>  MODULE_DEVICE_TABLE(platform, ad7606_driver_ids);
> @@ -134,6 +223,7 @@ static const struct of_device_id ad7606_of_match[] = {
>         { .compatible = "adi,ad7606-4", .data = &ad7606_4_info },
>         { .compatible = "adi,ad7606-6", .data = &ad7606_6_info },
>         { .compatible = "adi,ad7606-8", .data = &ad7606_8_info },
> +       { .compatible = "adi,ad7606b", .data = &ad7606b_info },
>         { }
>  };
>  MODULE_DEVICE_TABLE(of, ad7606_of_match);
> @@ -153,3 +243,4 @@ MODULE_AUTHOR("Michael Hennerich <michael.hennerich@analog.com>");
>  MODULE_DESCRIPTION("Analog Devices AD7606 ADC");
>  MODULE_LICENSE("GPL v2");
>  MODULE_IMPORT_NS(IIO_AD7606);
> +MODULE_IMPORT_NS(IIO_BACKEND);
>
> --
> 2.34.1
>ad7606b_bi_channels
Jonathan Cameron Sept. 29, 2024, 12:54 p.m. UTC | #2
On Fri, 20 Sep 2024 17:33:29 +0000
Guillaume Stols <gstols@baylibre.com> wrote:

> - Basic support for iio backend.
> - Supports IIO_CHAN_INFO_SAMP_FREQ R/W.
> - Only hardware mode is available, and that IIO_CHAN_INFO_RAW is not
>   supported if iio-backend mode is selected.
> 
> Signed-off-by: Guillaume Stols <gstols@baylibre.com>
One trivial comment to add to what David covered.
I'm also curious now on what the limitation is that meant we didn't
support the AD7606B with parallel interface before, and what stops us
supporting other devices with the backend IP?  Is there a fundamental difference?
> ---
>  drivers/iio/adc/Kconfig      |  2 +
>  drivers/iio/adc/ad7606.c     | 94 +++++++++++++++++++++++++++++++++++++-------
>  drivers/iio/adc/ad7606.h     | 15 +++++++
>  drivers/iio/adc/ad7606_par.c | 91 ++++++++++++++++++++++++++++++++++++++++++
>  4 files changed, 187 insertions(+), 15 deletions(-)
> 
> diff --git a/drivers/iio/adc/Kconfig b/drivers/iio/adc/Kconfig
> index 4ab1a3092d88..9b52d5b2c592 100644
> --- a/drivers/iio/adc/Kconfig
> +++ b/drivers/iio/adc/Kconfig
> @@ -224,9 +224,11 @@ config AD7606_IFACE_PARALLEL
>  	tristate "Analog Devices AD7606 ADC driver with parallel interface support"
>  	depends on HAS_IOPORT
>  	select AD7606
> +	select IIO_BACKEND
>  	help
>  	  Say yes here to build parallel interface support for Analog Devices:
>  	  ad7605-4, ad7606, ad7606-6, ad7606-4 analog to digital converters (ADC).
> +	  It also support iio_backended devices for AD7606B.

But not for other devices?  Is the expectation that they will need
a different IP for the backend, or just a case of not tested yet?


>  
>  	  To compile this driver as a module, choose M here: the
>  	  module will be called ad7606_par.

> +
> +static int ad7606_update_scan_mode(struct iio_dev *indio_dev,
> +				   const unsigned long *scan_mask)
> +{
> +	struct ad7606_state *st = iio_priv(indio_dev);
> +
> +	/* The update scan mode is only for iio backend compatible drivers.
	/*
	 * The update...

> +	 * If the specific update_scan_mode is not defined in the bus ops,
> +	 * just do nothing and return 0.
> +	 */
> +	if (st->bops->update_scan_mode)
> +		return st->bops->update_scan_mode(indio_dev, scan_mask);
> +	else
> +		return 0;
> +}
> +
diff mbox series

Patch

diff --git a/drivers/iio/adc/Kconfig b/drivers/iio/adc/Kconfig
index 4ab1a3092d88..9b52d5b2c592 100644
--- a/drivers/iio/adc/Kconfig
+++ b/drivers/iio/adc/Kconfig
@@ -224,9 +224,11 @@  config AD7606_IFACE_PARALLEL
 	tristate "Analog Devices AD7606 ADC driver with parallel interface support"
 	depends on HAS_IOPORT
 	select AD7606
+	select IIO_BACKEND
 	help
 	  Say yes here to build parallel interface support for Analog Devices:
 	  ad7605-4, ad7606, ad7606-6, ad7606-4 analog to digital converters (ADC).
+	  It also support iio_backended devices for AD7606B.
 
 	  To compile this driver as a module, choose M here: the
 	  module will be called ad7606_par.
diff --git a/drivers/iio/adc/ad7606.c b/drivers/iio/adc/ad7606.c
index 7f2ff1674638..f710445bdc22 100644
--- a/drivers/iio/adc/ad7606.c
+++ b/drivers/iio/adc/ad7606.c
@@ -21,6 +21,7 @@ 
 #include <linux/util_macros.h>
 #include <linux/units.h>
 
+#include <linux/iio/backend.h>
 #include <linux/iio/iio.h>
 #include <linux/iio/buffer.h>
 #include <linux/iio/sysfs.h>
@@ -271,7 +272,15 @@  static int ad7606_set_sampling_freq(struct ad7606_state *st, unsigned long freq)
 
 static int ad7606_read_samples(struct ad7606_state *st)
 {
-	unsigned int num = st->chip_info->num_channels - 1;
+	unsigned int num = st->chip_info->num_channels;
+
+	/*
+	 * Timestamp channel does not contain sample, and no timestamp channel if
+	 * backend is used.
+	 */
+	if (!st->back)
+		num--;
+
 	u16 *data = st->data;
 
 	return st->bops->read_block(st->dev, num, data);
@@ -319,11 +328,14 @@  static int ad7606_scan_direct(struct iio_dev *indio_dev, unsigned int ch)
 		if (!ret)
 			return ret;
 	}
-	ret = wait_for_completion_timeout(&st->completion,
-					  msecs_to_jiffies(1000));
-	if (!ret) {
-		ret = -ETIMEDOUT;
-		goto error_ret;
+
+	if (!st->back) {
+		ret = wait_for_completion_timeout(&st->completion,
+						  msecs_to_jiffies(1000));
+		if (!ret) {
+			ret = -ETIMEDOUT;
+			goto error_ret;
+		}
 	}
 
 	ret = ad7606_read_samples(st);
@@ -349,6 +361,7 @@  static int ad7606_read_raw(struct iio_dev *indio_dev,
 {
 	int ret, ch = 0;
 	struct ad7606_state *st = iio_priv(indio_dev);
+	struct pwm_state cnvst_pwm_state;
 
 	switch (m) {
 	case IIO_CHAN_INFO_RAW:
@@ -369,6 +382,10 @@  static int ad7606_read_raw(struct iio_dev *indio_dev,
 	case IIO_CHAN_INFO_OVERSAMPLING_RATIO:
 		*val = st->oversampling;
 		return IIO_VAL_INT;
+	case IIO_CHAN_INFO_SAMP_FREQ:
+		pwm_get_state(st->cnvst_pwm, &cnvst_pwm_state);
+		*val = DIV_ROUND_CLOSEST_ULL(NSEC_PER_SEC, cnvst_pwm_state.period);
+		return IIO_VAL_INT;
 	}
 	return -EINVAL;
 }
@@ -458,6 +475,8 @@  static int ad7606_write_raw(struct iio_dev *indio_dev,
 			return ret;
 
 		return 0;
+	case IIO_CHAN_INFO_SAMP_FREQ:
+		return ad7606_set_sampling_freq(st, val);
 	default:
 		return -EINVAL;
 	}
@@ -595,14 +614,49 @@  static int ad7606_buffer_predisable(struct iio_dev *indio_dev)
 	return 0;
 }
 
+static int ad7606_pwm_buffer_postenable(struct iio_dev *indio_dev)
+{
+	struct ad7606_state *st = iio_priv(indio_dev);
+
+	return ad7606_pwm_set_swing(st);
+}
+
+static int ad7606_pwm_buffer_predisable(struct iio_dev *indio_dev)
+{
+	struct ad7606_state *st = iio_priv(indio_dev);
+
+	return ad7606_pwm_set_low(st);
+}
+
+static int ad7606_update_scan_mode(struct iio_dev *indio_dev,
+				   const unsigned long *scan_mask)
+{
+	struct ad7606_state *st = iio_priv(indio_dev);
+
+	/* The update scan mode is only for iio backend compatible drivers.
+	 * If the specific update_scan_mode is not defined in the bus ops,
+	 * just do nothing and return 0.
+	 */
+	if (st->bops->update_scan_mode)
+		return st->bops->update_scan_mode(indio_dev, scan_mask);
+	else
+		return 0;
+}
+
 static const struct iio_buffer_setup_ops ad7606_buffer_ops = {
 	.postenable = &ad7606_buffer_postenable,
 	.predisable = &ad7606_buffer_predisable,
 };
 
+static const struct iio_buffer_setup_ops ad7606_pwm_buffer_ops = {
+	.postenable = &ad7606_pwm_buffer_postenable,
+	.predisable = &ad7606_pwm_buffer_predisable,
+};
+
 static const struct iio_info ad7606_info_no_os_or_range = {
 	.read_raw = &ad7606_read_raw,
 	.validate_trigger = &ad7606_validate_trigger,
+	.update_scan_mode = &ad7606_update_scan_mode,
 };
 
 static const struct iio_info ad7606_info_os_and_range = {
@@ -610,6 +664,7 @@  static const struct iio_info ad7606_info_os_and_range = {
 	.write_raw = &ad7606_write_raw,
 	.attrs = &ad7606_attribute_group_os_and_range,
 	.validate_trigger = &ad7606_validate_trigger,
+	.update_scan_mode = &ad7606_update_scan_mode,
 };
 
 static const struct iio_info ad7606_info_os_range_and_debug = {
@@ -618,6 +673,7 @@  static const struct iio_info ad7606_info_os_range_and_debug = {
 	.debugfs_reg_access = &ad7606_reg_access,
 	.attrs = &ad7606_attribute_group_os_and_range,
 	.validate_trigger = &ad7606_validate_trigger,
+	.update_scan_mode = &ad7606_update_scan_mode,
 };
 
 static const struct iio_info ad7606_info_os = {
@@ -625,6 +681,7 @@  static const struct iio_info ad7606_info_os = {
 	.write_raw = &ad7606_write_raw,
 	.attrs = &ad7606_attribute_group_os,
 	.validate_trigger = &ad7606_validate_trigger,
+	.update_scan_mode = &ad7606_update_scan_mode,
 };
 
 static const struct iio_info ad7606_info_range = {
@@ -632,6 +689,7 @@  static const struct iio_info ad7606_info_range = {
 	.write_raw = &ad7606_write_raw,
 	.attrs = &ad7606_attribute_group_range,
 	.validate_trigger = &ad7606_validate_trigger,
+	.update_scan_mode = &ad7606_update_scan_mode,
 };
 
 static const struct iio_trigger_ops ad7606_trigger_ops = {
@@ -700,8 +758,6 @@  int ad7606_probe(struct device *dev, int irq, void __iomem *base_address,
 	indio_dev->channels = st->chip_info->channels;
 	indio_dev->num_channels = st->chip_info->num_channels;
 
-	init_completion(&st->completion);
-
 	ret = ad7606_reset(st);
 	if (ret)
 		dev_warn(st->dev, "failed to RESET: no RESET GPIO specified\n");
@@ -774,14 +830,22 @@  int ad7606_probe(struct device *dev, int irq, void __iomem *base_address,
 		if (ret)
 			return ret;
 	}
-	ret = devm_request_threaded_irq(dev, irq,
-					NULL,
-					&ad7606_interrupt,
-					IRQF_TRIGGER_FALLING | IRQF_ONESHOT,
-					chip_info->name, indio_dev);
-	if (ret)
-		return ret;
 
+	if (st->bops->iio_backend_config) {
+		ret = st->bops->iio_backend_config(dev, indio_dev);
+		if (ret)
+			return ret;
+		indio_dev->setup_ops = &ad7606_pwm_buffer_ops;
+	} else {
+		init_completion(&st->completion);
+		ret = devm_request_threaded_irq(dev, irq,
+						NULL,
+						&ad7606_interrupt,
+						IRQF_TRIGGER_FALLING | IRQF_ONESHOT,
+						chip_info->name, indio_dev);
+		if (ret)
+			return ret;
+	}
 	return devm_iio_device_register(dev, indio_dev);
 }
 EXPORT_SYMBOL_NS_GPL(ad7606_probe, IIO_AD7606);
diff --git a/drivers/iio/adc/ad7606.h b/drivers/iio/adc/ad7606.h
index 18c87fe9a41a..53cd8eb4898e 100644
--- a/drivers/iio/adc/ad7606.h
+++ b/drivers/iio/adc/ad7606.h
@@ -34,6 +34,12 @@ 
 		BIT(IIO_CHAN_INFO_SCALE),		\
 		BIT(IIO_CHAN_INFO_OVERSAMPLING_RATIO))
 
+#define AD7606_BI_CHANNEL(num)				\
+	AD760X_CHANNEL(num, 0,				\
+		BIT(IIO_CHAN_INFO_SCALE),		\
+		BIT(IIO_CHAN_INFO_SAMP_FREQ) |		\
+		BIT(IIO_CHAN_INFO_OVERSAMPLING_RATIO))
+
 #define AD7616_CHANNEL(num)	\
 	AD760X_CHANNEL(num, BIT(IIO_CHAN_INFO_RAW) | BIT(IIO_CHAN_INFO_SCALE),\
 		0, BIT(IIO_CHAN_INFO_OVERSAMPLING_RATIO))
@@ -59,6 +65,7 @@  enum ad7606_supported_device_ids {
  * @os_req_reset	some devices require a reset to update oversampling
  * @init_delay_ms	required delay in miliseconds for initialization
  *			after a restart
+ * @has_backend		defines if a backend is available for the given chip
  */
 struct ad7606_chip_info {
 	enum ad7606_supported_device_ids id;
@@ -69,6 +76,7 @@  struct ad7606_chip_info {
 	unsigned int			oversampling_num;
 	bool				os_req_reset;
 	unsigned long			init_delay_ms;
+	bool				has_backend;
 };
 
 /**
@@ -115,6 +123,7 @@  struct ad7606_state {
 	unsigned int			num_scales;
 	const unsigned int		*oversampling_avail;
 	unsigned int			num_os_ratios;
+	struct iio_backend		*back;
 	int (*write_scale)(struct iio_dev *indio_dev, int ch, int val);
 	int (*write_os)(struct iio_dev *indio_dev, int val);
 
@@ -139,16 +148,21 @@  struct ad7606_state {
 
 /**
  * struct ad7606_bus_ops - driver bus operations
+ * @iio_backend_config	function pointer for configuring the iio_backend for
+ *			the compatibles that use it
  * @read_block		function pointer for reading blocks of data
  * @sw_mode_config:	pointer to a function which configured the device
  *			for software mode
  * @reg_read	function pointer for reading spi register
  * @reg_write	function pointer for writing spi register
  * @write_mask	function pointer for write spi register with mask
+ * @update_scan_mode	function pointer for handling the calls to iio_info's update_scan
+ *			mode when enabling/disabling channels.
  * @rd_wr_cmd	pointer to the function which calculates the spi address
  */
 struct ad7606_bus_ops {
 	/* more methods added in future? */
+	int (*iio_backend_config)(struct device *dev, struct iio_dev *indio_dev);
 	int (*read_block)(struct device *dev, int num, void *data);
 	int (*sw_mode_config)(struct iio_dev *indio_dev);
 	int (*reg_read)(struct ad7606_state *st, unsigned int addr);
@@ -159,6 +173,7 @@  struct ad7606_bus_ops {
 				 unsigned int addr,
 				 unsigned long mask,
 				 unsigned int val);
+	int (*update_scan_mode)(struct iio_dev *indio_dev, const unsigned long *scan_mask);
 	u16 (*rd_wr_cmd)(int addr, char isWriteOp);
 };
 
diff --git a/drivers/iio/adc/ad7606_par.c b/drivers/iio/adc/ad7606_par.c
index 7bac39033955..564284ede997 100644
--- a/drivers/iio/adc/ad7606_par.c
+++ b/drivers/iio/adc/ad7606_par.c
@@ -3,6 +3,8 @@ 
  * AD7606 Parallel Interface ADC driver
  *
  * Copyright 2011 Analog Devices Inc.
+ * Copyright 2024 Analog Devices Inc.
+ * Copyright 2024 BayLibre SAS.
  */
 
 #include <linux/err.h>
@@ -15,8 +17,80 @@ 
 #include <linux/types.h>
 
 #include <linux/iio/iio.h>
+#include <linux/iio/backend.h>
+
 #include "ad7606.h"
 
+static const struct iio_chan_spec ad7606b_bi_channels[] = {
+	AD7606_BI_CHANNEL(0),
+	AD7606_BI_CHANNEL(1),
+	AD7606_BI_CHANNEL(2),
+	AD7606_BI_CHANNEL(3),
+	AD7606_BI_CHANNEL(4),
+	AD7606_BI_CHANNEL(5),
+	AD7606_BI_CHANNEL(6),
+	AD7606_BI_CHANNEL(7),
+};
+
+static int ad7606_bi_update_scan_mode(struct iio_dev *indio_dev, const unsigned long *scan_mask)
+{
+	struct ad7606_state *st = iio_priv(indio_dev);
+	unsigned int c, ret;
+
+	for (c = 0; c < indio_dev->num_channels; c++) {
+		if (test_bit(c, scan_mask))
+			ret = iio_backend_chan_enable(st->back, c);
+		else
+			ret = iio_backend_chan_disable(st->back, c);
+		if (ret)
+			return ret;
+	}
+
+	return 0;
+}
+
+static int ad7606_bi_setup_iio_backend(struct device *dev, struct iio_dev *indio_dev)
+{
+	struct ad7606_state *st = iio_priv(indio_dev);
+	unsigned int ret, c;
+	struct iio_backend_data_fmt data = {
+		.sign_extend = true,
+		.enable = true,
+	};
+
+	st->back = devm_iio_backend_get(dev, NULL);
+	if (IS_ERR(st->back))
+		return PTR_ERR(st->back);
+
+	/* If the device is iio_backend powered the PWM is mandatory */
+	if (!st->cnvst_pwm)
+		return -EINVAL;
+
+	ret = devm_iio_backend_request_buffer(dev, st->back, indio_dev);
+	if (ret)
+		return ret;
+
+	ret = devm_iio_backend_enable(dev, st->back);
+	if (ret)
+		return ret;
+
+	for (c = 0; c < indio_dev->num_channels; c++) {
+		ret = iio_backend_data_format_set(st->back, c, &data);
+		if (ret)
+			return ret;
+	}
+
+	indio_dev->channels = ad7606b_bi_channels;
+	indio_dev->num_channels = 8;
+
+	return 0;
+}
+
+static const struct ad7606_bus_ops ad7606_bi_bops = {
+	.iio_backend_config = ad7606_bi_setup_iio_backend,
+	.update_scan_mode = ad7606_bi_update_scan_mode,
+};
+
 static int ad7606_par16_read_block(struct device *dev,
 				   int count, void *buf)
 {
@@ -96,9 +170,23 @@  static int ad7606_par_probe(struct platform_device *pdev)
 	void __iomem *addr;
 	resource_size_t remap_size;
 	int irq;
+	struct iio_backend *back;
 
+	/*
+	 * If a firmware node is available (ACPI or DT), platform_device_id is null
+	 * and we must use get_match_data.
+	 */
 	if (dev_fwnode(&pdev->dev)) {
 		chip_info = device_get_match_data(&pdev->dev);
+		back = devm_iio_backend_get(&pdev->dev, NULL);
+		if (!IS_ERR(back))
+			/*
+			 * If a backend is available in the device tree, call the core
+			 * probe with backend bops, otherwise use the former bops.
+			 */
+			return ad7606_probe(&pdev->dev, 0, NULL,
+					    chip_info,
+					    &ad7606_bi_bops);
 	} else {
 		id = platform_get_device_id(pdev);
 		chip_info = (const struct ad7606_chip_info *)id->driver_data;
@@ -125,6 +213,7 @@  static const struct platform_device_id ad7606_driver_ids[] = {
 	{ .name	= "ad7606-4", .driver_data = (kernel_ulong_t)&ad7606_4_info, },
 	{ .name	= "ad7606-6", .driver_data = (kernel_ulong_t)&ad7606_6_info, },
 	{ .name	= "ad7606-8", .driver_data = (kernel_ulong_t)&ad7606_8_info, },
+	{ .name	= "ad7606b", .driver_data = (kernel_ulong_t)&ad7606b_info, },
 	{ }
 };
 MODULE_DEVICE_TABLE(platform, ad7606_driver_ids);
@@ -134,6 +223,7 @@  static const struct of_device_id ad7606_of_match[] = {
 	{ .compatible = "adi,ad7606-4", .data = &ad7606_4_info },
 	{ .compatible = "adi,ad7606-6", .data = &ad7606_6_info },
 	{ .compatible = "adi,ad7606-8", .data = &ad7606_8_info },
+	{ .compatible = "adi,ad7606b", .data = &ad7606b_info },
 	{ }
 };
 MODULE_DEVICE_TABLE(of, ad7606_of_match);
@@ -153,3 +243,4 @@  MODULE_AUTHOR("Michael Hennerich <michael.hennerich@analog.com>");
 MODULE_DESCRIPTION("Analog Devices AD7606 ADC");
 MODULE_LICENSE("GPL v2");
 MODULE_IMPORT_NS(IIO_AD7606);
+MODULE_IMPORT_NS(IIO_BACKEND);