* [PATCH v3 0/4] iio: adc: ad4000: Add SPI offload support
@ 2025-03-26 13:24 Marcelo Schmitt
2025-03-26 13:24 ` [PATCH v3 1/4] iio: adc: ad4000: Add support for SPI offload Marcelo Schmitt
` (3 more replies)
0 siblings, 4 replies; 12+ messages in thread
From: Marcelo Schmitt @ 2025-03-26 13:24 UTC (permalink / raw)
To: linux-iio, linux-doc, linux-kernel
Cc: jic23, lars, Michael.Hennerich, corbet, dlechner,
marcelo.schmitt1
This patch series extends the ad4000 driver to support SPI offloading.
In addition to that, ad4000 IIO documentation is expanded to:
- list PulSAR parts supported by the ad4000 driver.
- describe some characteristics of AD4000 IIO device.
- describe changes when SPI offload is being used.
Change log v2 -> v3
- Dropped patch 1 of v2 series and reverted buffer endianness and SPI transfer
bits_per_word configurations to the way they were in v1.
- Removed dummy transfer in ad4000_prepare_offload_turbo_message().
- Unified offload message preparation.
- Removed spi->cs_hold.value to use cs_inactive to set CNV/CS high time.
- Adjusted max sample rate for AD7983 and AD7984.
- Dropped AD4000_TCONV_NS and made all time related constants a time_spec field.
- Removed _offl param from _CHANNELS() macros and added a comment about that.
- Commented IIO_BE check in ad4000_prepare_3wire_mode_message().
- Now using predisable to balance with postenable.
- Put spi/offload/consumer.h include right after spi/spi.h.
- Removed unrelated brackets in info_mask_separate_available assignment
- Added blank line after if.
Change log v1 -> v2
- Fixed passing inappropriate pointer instead of ret to dev_err_probe()
- [new patch] Set transfer bits_per_word to have data in CPU endianness
- Set iio_dev num_channels close to where channels is set.
- Complement offload message comment about first sample being invalid
- Document why the first buffer sample is invalid when offloading
- Added blank line before a 'simple return'.
Link to v2: https://lore.kernel.org/linux-iio/cover.1742394806.git.marcelo.schmitt@analog.com/
Link to v1: https://lore.kernel.org/linux-iio/cover.1741970538.git.marcelo.schmitt@analog.com/
v3 was tested and worked with the following setups
- ADAQ4003 on CoraZ7 with SPI offload support;
- AD7687 on CoraZ7 with SPI offload support;
- AD7687 on rpi (no SPI offload support).
The Linux image ran on cora was built from IIO testing branch.
The Linux image ran on rpi was built from rpi tree rpi-6.14.y branch with
some adaptations to cope with old iio_device_claim_direct_mode() and addition
of the following patches
iio: adc: ad4000: Stop using iio_device_claim_direct_scoped()
spi: add basic support for SPI offloading
spi: offload: add support for hardware triggers
dt-bindings: trigger-source: add generic PWM trigger source
spi: offload-trigger: add PWM trigger driver
spi: add offload TX/RX streaming APIs
iio: buffer-dmaengine: split requesting DMA channel from allocating buffer
iio: buffer-dmaengine: add devm_iio_dmaengine_buffer_setup_with_handle()
Marcelo Schmitt (4):
iio: adc: ad4000: Add support for SPI offload
Documentation: iio: ad4000: Add new supported parts
Documentation: iio: ad4000: Add IIO Device characteristics section
Documentation: iio: ad4000: Describe offload support
Documentation/iio/ad4000.rst | 86 +++++++-
drivers/iio/adc/Kconfig | 7 +-
drivers/iio/adc/ad4000.c | 405 ++++++++++++++++++++++++++++++++---
3 files changed, 463 insertions(+), 35 deletions(-)
base-commit: af94f401e26f686f7391ce79b38a6129417c22dc
prerequisite-patch-id: 3d517eef53a799adba5922815fe684b913e36773
--
2.47.2
^ permalink raw reply [flat|nested] 12+ messages in thread* [PATCH v3 1/4] iio: adc: ad4000: Add support for SPI offload 2025-03-26 13:24 [PATCH v3 0/4] iio: adc: ad4000: Add SPI offload support Marcelo Schmitt @ 2025-03-26 13:24 ` Marcelo Schmitt 2025-03-27 16:12 ` David Lechner 2025-03-26 13:25 ` [PATCH v3 2/4] Documentation: iio: ad4000: Add new supported parts Marcelo Schmitt ` (2 subsequent siblings) 3 siblings, 1 reply; 12+ messages in thread From: Marcelo Schmitt @ 2025-03-26 13:24 UTC (permalink / raw) To: linux-iio, linux-doc, linux-kernel Cc: jic23, lars, Michael.Hennerich, corbet, dlechner, marcelo.schmitt1 FPGA HDL projects can include a PWM generator in addition to SPI-Engine. The PWM IP is used to trigger SPI-Engine offload modules that in turn set SPI-Engine to execute transfers to poll data from the ADC. That allows data to be read at the maximum sample rates. Also, it is possible to set a specific sample rate by setting the proper PWM duty cycle and related state parameters, thus allowing an adjustable ADC sample rate when a PWM (offload trigger) is used in combination with SPI-Engine. Add support for SPI offload. Signed-off-by: Marcelo Schmitt <marcelo.schmitt@analog.com> --- Ideally, we should do something to provide tquiet1 (if using offload) or tconv (is not using turbo mode) delay but, when testing with AD7687 and ADAQ4003 on cora with offload support, I saw no issue in running the transfers without the CS/CNV high delay. Without the preparatory/dummy transfer, offload messages became equal so I dropped one of them. Also, despite SPI-Engine not supporting it, I set cs_inactive which semantic seems to match the CS high delay we want for these transfers. drivers/iio/adc/Kconfig | 7 +- drivers/iio/adc/ad4000.c | 405 +++++++++++++++++++++++++++++++++++---- 2 files changed, 378 insertions(+), 34 deletions(-) diff --git a/drivers/iio/adc/Kconfig b/drivers/iio/adc/Kconfig index b7ae6e0ae0df..1cfa3a32f3a7 100644 --- a/drivers/iio/adc/Kconfig +++ b/drivers/iio/adc/Kconfig @@ -25,10 +25,15 @@ config AD4000 tristate "Analog Devices AD4000 ADC Driver" depends on SPI select IIO_BUFFER + select IIO_BUFFER_DMAENGINE select IIO_TRIGGERED_BUFFER + select SPI_OFFLOAD help Say yes here to build support for Analog Devices AD4000 high speed - SPI analog to digital converters (ADC). + SPI analog to digital converters (ADC). If intended to use with + SPI offloading support, it is recommended to enable + CONFIG_SPI_AXI_SPI_ENGINE, CONFIG_PWM_AXI_PWMGEN, and + CONFIG_SPI_OFFLOAD_TRIGGER_PWM. To compile this driver as a module, choose M here: the module will be called ad4000. diff --git a/drivers/iio/adc/ad4000.c b/drivers/iio/adc/ad4000.c index 4fe8dee48da9..9fc56853265e 100644 --- a/drivers/iio/adc/ad4000.c +++ b/drivers/iio/adc/ad4000.c @@ -16,11 +16,13 @@ #include <linux/gpio/consumer.h> #include <linux/regulator/consumer.h> #include <linux/spi/spi.h> +#include <linux/spi/offload/consumer.h> #include <linux/units.h> #include <linux/util_macros.h> #include <linux/iio/iio.h> #include <linux/iio/buffer.h> +#include <linux/iio/buffer-dmaengine.h> #include <linux/iio/triggered_buffer.h> #include <linux/iio/trigger_consumer.h> @@ -32,10 +34,11 @@ /* AD4000 Configuration Register programmable bits */ #define AD4000_CFG_SPAN_COMP BIT(3) /* Input span compression */ #define AD4000_CFG_HIGHZ BIT(2) /* High impedance mode */ +#define AD4000_CFG_TURBO BIT(1) /* Turbo mode */ #define AD4000_SCALE_OPTIONS 2 -#define __AD4000_DIFF_CHANNEL(_sign, _real_bits, _storage_bits, _reg_access) \ +#define __AD4000_DIFF_CHANNEL(_sign, _real_bits, _storage_bits, _reg_access, _offl)\ { \ .type = IIO_VOLTAGE, \ .indexed = 1, \ @@ -43,54 +46,65 @@ .channel = 0, \ .channel2 = 1, \ .info_mask_separate = BIT(IIO_CHAN_INFO_RAW) | \ - BIT(IIO_CHAN_INFO_SCALE), \ + BIT(IIO_CHAN_INFO_SCALE) | \ + (_offl ? BIT(IIO_CHAN_INFO_SAMP_FREQ) : 0), \ .info_mask_separate_available = _reg_access ? BIT(IIO_CHAN_INFO_SCALE) : 0,\ .scan_index = 0, \ .scan_type = { \ .sign = _sign, \ .realbits = _real_bits, \ .storagebits = _storage_bits, \ - .shift = _storage_bits - _real_bits, \ - .endianness = IIO_BE, \ + .shift = (_offl ? 0 : _storage_bits - _real_bits), \ + .endianness = _offl ? IIO_CPU : IIO_BE \ }, \ } -#define AD4000_DIFF_CHANNEL(_sign, _real_bits, _reg_access) \ +#define AD4000_DIFF_CHANNEL(_sign, _real_bits, _reg_access, _offl) \ __AD4000_DIFF_CHANNEL((_sign), (_real_bits), \ - ((_real_bits) > 16 ? 32 : 16), (_reg_access)) + (((_offl) || ((_real_bits) > 16)) ? 32 : 16), \ + (_reg_access), (_offl)) +/* + * When SPI offload is configured, transfers are executed withouth CPU + * intervention so no soft timestamp can be recorded when transfers run. + * Because of that, the macros that set timestamp channel are only used when + * transfers are not offloaded. + */ #define AD4000_DIFF_CHANNELS(_sign, _real_bits, _reg_access) \ { \ - AD4000_DIFF_CHANNEL(_sign, _real_bits, _reg_access), \ + AD4000_DIFF_CHANNEL(_sign, _real_bits, _reg_access, 0), \ IIO_CHAN_SOFT_TIMESTAMP(1), \ } -#define __AD4000_PSEUDO_DIFF_CHANNEL(_sign, _real_bits, _storage_bits, _reg_access)\ +#define __AD4000_PSEUDO_DIFF_CHANNEL(_sign, _real_bits, _storage_bits, \ + _reg_access, _offl) \ { \ .type = IIO_VOLTAGE, \ .indexed = 1, \ .channel = 0, \ .info_mask_separate = BIT(IIO_CHAN_INFO_RAW) | \ BIT(IIO_CHAN_INFO_SCALE) | \ - BIT(IIO_CHAN_INFO_OFFSET), \ + BIT(IIO_CHAN_INFO_OFFSET) | \ + (_offl ? BIT(IIO_CHAN_INFO_SAMP_FREQ) : 0), \ .info_mask_separate_available = _reg_access ? BIT(IIO_CHAN_INFO_SCALE) : 0,\ .scan_index = 0, \ .scan_type = { \ .sign = _sign, \ .realbits = _real_bits, \ .storagebits = _storage_bits, \ - .shift = _storage_bits - _real_bits, \ - .endianness = IIO_BE, \ + .shift = (_offl ? 0 : _storage_bits - _real_bits), \ + .endianness = _offl ? IIO_CPU : IIO_BE \ }, \ } -#define AD4000_PSEUDO_DIFF_CHANNEL(_sign, _real_bits, _reg_access) \ +#define AD4000_PSEUDO_DIFF_CHANNEL(_sign, _real_bits, _reg_access, _offl) \ __AD4000_PSEUDO_DIFF_CHANNEL((_sign), (_real_bits), \ - ((_real_bits) > 16 ? 32 : 16), (_reg_access)) + (((_offl) || ((_real_bits) > 16)) ? 32 : 16),\ + (_reg_access), (_offl)) #define AD4000_PSEUDO_DIFF_CHANNELS(_sign, _real_bits, _reg_access) \ { \ - AD4000_PSEUDO_DIFF_CHANNEL(_sign, _real_bits, _reg_access), \ + AD4000_PSEUDO_DIFF_CHANNEL(_sign, _real_bits, _reg_access, 0), \ IIO_CHAN_SOFT_TIMESTAMP(1), \ } @@ -120,6 +134,7 @@ static const int ad4000_gains[] = { struct ad4000_time_spec { int t_conv_ns; + int t_quiet1_ns; int t_quiet2_ns; }; @@ -129,54 +144,63 @@ struct ad4000_time_spec { */ static const struct ad4000_time_spec ad4000_t_spec = { .t_conv_ns = 320, + .t_quiet1_ns = 190, .t_quiet2_ns = 60, }; /* AD4020, AD4021, AD4022 */ static const struct ad4000_time_spec ad4020_t_spec = { .t_conv_ns = 350, + .t_quiet1_ns = 200, .t_quiet2_ns = 60, }; /* AD7983, AD7984 */ static const struct ad4000_time_spec ad7983_t_spec = { .t_conv_ns = 500, + .t_quiet1_ns = 0, .t_quiet2_ns = 0, }; /* AD7980, AD7982 */ static const struct ad4000_time_spec ad7980_t_spec = { .t_conv_ns = 800, + .t_quiet1_ns = 0, .t_quiet2_ns = 0, }; /* AD7946, AD7686, AD7688, AD7988-5, AD7693 */ static const struct ad4000_time_spec ad7686_t_spec = { .t_conv_ns = 1600, + .t_quiet1_ns = 0, .t_quiet2_ns = 0, }; /* AD7690 */ static const struct ad4000_time_spec ad7690_t_spec = { .t_conv_ns = 2100, + .t_quiet1_ns = 0, .t_quiet2_ns = 0, }; /* AD7942, AD7685, AD7687 */ static const struct ad4000_time_spec ad7687_t_spec = { .t_conv_ns = 3200, + .t_quiet1_ns = 0, .t_quiet2_ns = 0, }; /* AD7691 */ static const struct ad4000_time_spec ad7691_t_spec = { .t_conv_ns = 3700, + .t_quiet1_ns = 0, .t_quiet2_ns = 0, }; /* AD7988-1 */ static const struct ad4000_time_spec ad7988_1_t_spec = { .t_conv_ns = 9500, + .t_quiet1_ns = 0, .t_quiet2_ns = 0, }; @@ -184,212 +208,299 @@ struct ad4000_chip_info { const char *dev_name; struct iio_chan_spec chan_spec[2]; struct iio_chan_spec reg_access_chan_spec[2]; + struct iio_chan_spec offload_chan_spec; + struct iio_chan_spec reg_access_offload_chan_spec; const struct ad4000_time_spec *time_spec; bool has_hardware_gain; + int max_rate_hz; }; static const struct ad4000_chip_info ad4000_chip_info = { .dev_name = "ad4000", .chan_spec = AD4000_PSEUDO_DIFF_CHANNELS('u', 16, 0), .reg_access_chan_spec = AD4000_PSEUDO_DIFF_CHANNELS('u', 16, 1), + .offload_chan_spec = AD4000_PSEUDO_DIFF_CHANNEL('u', 16, 0, 1), + .reg_access_offload_chan_spec = AD4000_PSEUDO_DIFF_CHANNEL('u', 16, 1, 1), .time_spec = &ad4000_t_spec, + .max_rate_hz = 2 * MEGA, }; static const struct ad4000_chip_info ad4001_chip_info = { .dev_name = "ad4001", .chan_spec = AD4000_DIFF_CHANNELS('s', 16, 0), .reg_access_chan_spec = AD4000_DIFF_CHANNELS('s', 16, 1), + .offload_chan_spec = AD4000_DIFF_CHANNEL('s', 16, 0, 1), + .reg_access_offload_chan_spec = AD4000_DIFF_CHANNEL('s', 16, 1, 1), .time_spec = &ad4000_t_spec, + .max_rate_hz = 2 * MEGA, }; static const struct ad4000_chip_info ad4002_chip_info = { .dev_name = "ad4002", .chan_spec = AD4000_PSEUDO_DIFF_CHANNELS('u', 18, 0), .reg_access_chan_spec = AD4000_PSEUDO_DIFF_CHANNELS('u', 18, 1), + .offload_chan_spec = AD4000_PSEUDO_DIFF_CHANNEL('u', 18, 0, 1), + .reg_access_offload_chan_spec = AD4000_PSEUDO_DIFF_CHANNEL('u', 18, 1, 1), .time_spec = &ad4000_t_spec, + .max_rate_hz = 2 * MEGA, }; static const struct ad4000_chip_info ad4003_chip_info = { .dev_name = "ad4003", .chan_spec = AD4000_DIFF_CHANNELS('s', 18, 0), .reg_access_chan_spec = AD4000_DIFF_CHANNELS('s', 18, 1), + .offload_chan_spec = AD4000_DIFF_CHANNEL('s', 18, 0, 1), + .reg_access_offload_chan_spec = AD4000_DIFF_CHANNEL('s', 18, 1, 1), .time_spec = &ad4000_t_spec, + .max_rate_hz = 2 * MEGA, }; static const struct ad4000_chip_info ad4004_chip_info = { .dev_name = "ad4004", .chan_spec = AD4000_PSEUDO_DIFF_CHANNELS('u', 16, 0), .reg_access_chan_spec = AD4000_PSEUDO_DIFF_CHANNELS('u', 16, 1), + .offload_chan_spec = AD4000_PSEUDO_DIFF_CHANNEL('u', 16, 0, 1), + .reg_access_offload_chan_spec = AD4000_PSEUDO_DIFF_CHANNEL('u', 16, 1, 1), .time_spec = &ad4000_t_spec, + .max_rate_hz = 1 * MEGA, }; static const struct ad4000_chip_info ad4005_chip_info = { .dev_name = "ad4005", .chan_spec = AD4000_DIFF_CHANNELS('s', 16, 0), .reg_access_chan_spec = AD4000_DIFF_CHANNELS('s', 16, 1), + .offload_chan_spec = AD4000_DIFF_CHANNEL('s', 16, 0, 1), + .reg_access_offload_chan_spec = AD4000_DIFF_CHANNEL('s', 16, 1, 1), .time_spec = &ad4000_t_spec, + .max_rate_hz = 1 * MEGA, }; static const struct ad4000_chip_info ad4006_chip_info = { .dev_name = "ad4006", .chan_spec = AD4000_PSEUDO_DIFF_CHANNELS('u', 18, 0), .reg_access_chan_spec = AD4000_PSEUDO_DIFF_CHANNELS('u', 18, 1), + .offload_chan_spec = AD4000_PSEUDO_DIFF_CHANNEL('u', 18, 0, 1), + .reg_access_offload_chan_spec = AD4000_PSEUDO_DIFF_CHANNEL('u', 18, 1, 1), .time_spec = &ad4000_t_spec, + .max_rate_hz = 1 * MEGA, }; static const struct ad4000_chip_info ad4007_chip_info = { .dev_name = "ad4007", .chan_spec = AD4000_DIFF_CHANNELS('s', 18, 0), .reg_access_chan_spec = AD4000_DIFF_CHANNELS('s', 18, 1), + .offload_chan_spec = AD4000_DIFF_CHANNEL('s', 18, 0, 1), + .reg_access_offload_chan_spec = AD4000_DIFF_CHANNEL('s', 18, 1, 1), .time_spec = &ad4000_t_spec, + .max_rate_hz = 1 * MEGA, }; static const struct ad4000_chip_info ad4008_chip_info = { .dev_name = "ad4008", .chan_spec = AD4000_PSEUDO_DIFF_CHANNELS('u', 16, 0), .reg_access_chan_spec = AD4000_PSEUDO_DIFF_CHANNELS('u', 16, 1), + .offload_chan_spec = AD4000_PSEUDO_DIFF_CHANNEL('u', 16, 0, 1), + .reg_access_offload_chan_spec = AD4000_PSEUDO_DIFF_CHANNEL('u', 16, 1, 1), .time_spec = &ad4000_t_spec, + .max_rate_hz = 500 * KILO, }; static const struct ad4000_chip_info ad4010_chip_info = { .dev_name = "ad4010", .chan_spec = AD4000_PSEUDO_DIFF_CHANNELS('u', 18, 0), .reg_access_chan_spec = AD4000_PSEUDO_DIFF_CHANNELS('u', 18, 1), + .offload_chan_spec = AD4000_PSEUDO_DIFF_CHANNEL('u', 18, 0, 1), + .reg_access_offload_chan_spec = AD4000_PSEUDO_DIFF_CHANNEL('u', 18, 1, 1), .time_spec = &ad4000_t_spec, + .max_rate_hz = 500 * KILO, }; static const struct ad4000_chip_info ad4011_chip_info = { .dev_name = "ad4011", .chan_spec = AD4000_DIFF_CHANNELS('s', 18, 0), .reg_access_chan_spec = AD4000_DIFF_CHANNELS('s', 18, 1), + .offload_chan_spec = AD4000_DIFF_CHANNEL('s', 18, 0, 1), + .reg_access_offload_chan_spec = AD4000_DIFF_CHANNEL('s', 18, 1, 1), .time_spec = &ad4000_t_spec, + .max_rate_hz = 500 * KILO, }; static const struct ad4000_chip_info ad4020_chip_info = { .dev_name = "ad4020", .chan_spec = AD4000_DIFF_CHANNELS('s', 20, 0), .reg_access_chan_spec = AD4000_DIFF_CHANNELS('s', 20, 1), + .offload_chan_spec = AD4000_DIFF_CHANNEL('s', 20, 0, 1), + .reg_access_offload_chan_spec = AD4000_DIFF_CHANNEL('s', 20, 1, 1), .time_spec = &ad4020_t_spec, + .max_rate_hz = 1800 * KILO, }; static const struct ad4000_chip_info ad4021_chip_info = { .dev_name = "ad4021", .chan_spec = AD4000_DIFF_CHANNELS('s', 20, 0), .reg_access_chan_spec = AD4000_DIFF_CHANNELS('s', 20, 1), + .offload_chan_spec = AD4000_DIFF_CHANNEL('s', 20, 0, 1), + .reg_access_offload_chan_spec = AD4000_DIFF_CHANNEL('s', 20, 1, 1), .time_spec = &ad4020_t_spec, + .max_rate_hz = 1 * MEGA, }; static const struct ad4000_chip_info ad4022_chip_info = { .dev_name = "ad4022", .chan_spec = AD4000_DIFF_CHANNELS('s', 20, 0), .reg_access_chan_spec = AD4000_DIFF_CHANNELS('s', 20, 1), + .offload_chan_spec = AD4000_DIFF_CHANNEL('s', 20, 0, 1), + .reg_access_offload_chan_spec = AD4000_DIFF_CHANNEL('s', 20, 1, 1), .time_spec = &ad4020_t_spec, + .max_rate_hz = 500 * KILO, }; static const struct ad4000_chip_info adaq4001_chip_info = { .dev_name = "adaq4001", .chan_spec = AD4000_DIFF_CHANNELS('s', 16, 0), .reg_access_chan_spec = AD4000_DIFF_CHANNELS('s', 16, 1), + .offload_chan_spec = AD4000_DIFF_CHANNEL('s', 16, 0, 1), + .reg_access_offload_chan_spec = AD4000_DIFF_CHANNEL('s', 16, 1, 1), .time_spec = &ad4000_t_spec, .has_hardware_gain = true, + .max_rate_hz = 2 * MEGA, }; static const struct ad4000_chip_info adaq4003_chip_info = { .dev_name = "adaq4003", .chan_spec = AD4000_DIFF_CHANNELS('s', 18, 0), .reg_access_chan_spec = AD4000_DIFF_CHANNELS('s', 18, 1), + .offload_chan_spec = AD4000_DIFF_CHANNEL('s', 18, 0, 1), + .reg_access_offload_chan_spec = AD4000_DIFF_CHANNEL('s', 18, 1, 1), .time_spec = &ad4000_t_spec, .has_hardware_gain = true, + .max_rate_hz = 2 * MEGA, }; static const struct ad4000_chip_info ad7685_chip_info = { .dev_name = "ad7685", .chan_spec = AD4000_PSEUDO_DIFF_CHANNELS('u', 16, 0), + .offload_chan_spec = AD4000_PSEUDO_DIFF_CHANNEL('u', 16, 0, 1), .time_spec = &ad7687_t_spec, + .max_rate_hz = 250 * KILO, }; static const struct ad4000_chip_info ad7686_chip_info = { .dev_name = "ad7686", .chan_spec = AD4000_PSEUDO_DIFF_CHANNELS('u', 16, 0), + .offload_chan_spec = AD4000_PSEUDO_DIFF_CHANNEL('u', 16, 0, 1), .time_spec = &ad7686_t_spec, + .max_rate_hz = 500 * KILO, }; static const struct ad4000_chip_info ad7687_chip_info = { .dev_name = "ad7687", .chan_spec = AD4000_DIFF_CHANNELS('s', 16, 0), + .offload_chan_spec = AD4000_DIFF_CHANNEL('s', 16, 0, 1), .time_spec = &ad7687_t_spec, + .max_rate_hz = 250 * KILO, }; static const struct ad4000_chip_info ad7688_chip_info = { .dev_name = "ad7688", .chan_spec = AD4000_DIFF_CHANNELS('s', 16, 0), + .offload_chan_spec = AD4000_DIFF_CHANNEL('s', 16, 0, 1), .time_spec = &ad7686_t_spec, + .max_rate_hz = 500 * KILO, }; static const struct ad4000_chip_info ad7690_chip_info = { .dev_name = "ad7690", .chan_spec = AD4000_DIFF_CHANNELS('s', 18, 0), + .offload_chan_spec = AD4000_DIFF_CHANNEL('s', 18, 0, 1), .time_spec = &ad7690_t_spec, + .max_rate_hz = 400 * KILO, }; static const struct ad4000_chip_info ad7691_chip_info = { .dev_name = "ad7691", .chan_spec = AD4000_DIFF_CHANNELS('s', 18, 0), + .offload_chan_spec = AD4000_DIFF_CHANNEL('s', 18, 0, 1), .time_spec = &ad7691_t_spec, + .max_rate_hz = 250 * KILO, }; static const struct ad4000_chip_info ad7693_chip_info = { .dev_name = "ad7693", .chan_spec = AD4000_DIFF_CHANNELS('s', 16, 0), + .offload_chan_spec = AD4000_DIFF_CHANNEL('s', 16, 0, 1), .time_spec = &ad7686_t_spec, + .max_rate_hz = 500 * KILO, }; static const struct ad4000_chip_info ad7942_chip_info = { .dev_name = "ad7942", .chan_spec = AD4000_PSEUDO_DIFF_CHANNELS('u', 14, 0), + .offload_chan_spec = AD4000_PSEUDO_DIFF_CHANNEL('u', 14, 0, 1), .time_spec = &ad7687_t_spec, + .max_rate_hz = 250 * KILO, }; static const struct ad4000_chip_info ad7946_chip_info = { .dev_name = "ad7946", .chan_spec = AD4000_PSEUDO_DIFF_CHANNELS('u', 14, 0), + .offload_chan_spec = AD4000_PSEUDO_DIFF_CHANNEL('u', 14, 0, 1), .time_spec = &ad7686_t_spec, + .max_rate_hz = 500 * KILO, }; static const struct ad4000_chip_info ad7980_chip_info = { .dev_name = "ad7980", .chan_spec = AD4000_PSEUDO_DIFF_CHANNELS('u', 16, 0), + .offload_chan_spec = AD4000_PSEUDO_DIFF_CHANNEL('u', 16, 0, 1), .time_spec = &ad7980_t_spec, + .max_rate_hz = 1 * MEGA, }; static const struct ad4000_chip_info ad7982_chip_info = { .dev_name = "ad7982", .chan_spec = AD4000_DIFF_CHANNELS('s', 18, 0), + .offload_chan_spec = AD4000_DIFF_CHANNEL('s', 18, 0, 1), .time_spec = &ad7980_t_spec, + .max_rate_hz = 1 * MEGA, }; static const struct ad4000_chip_info ad7983_chip_info = { .dev_name = "ad7983", .chan_spec = AD4000_PSEUDO_DIFF_CHANNELS('u', 16, 0), + .offload_chan_spec = AD4000_PSEUDO_DIFF_CHANNEL('u', 16, 0, 1), .time_spec = &ad7983_t_spec, + .max_rate_hz = 1 * MEGA + 333 * KILO + 333, + }; static const struct ad4000_chip_info ad7984_chip_info = { .dev_name = "ad7984", .chan_spec = AD4000_DIFF_CHANNELS('s', 18, 0), + .offload_chan_spec = AD4000_DIFF_CHANNEL('s', 18, 0, 1), .time_spec = &ad7983_t_spec, + .max_rate_hz = 1 * MEGA + 333 * KILO + 333, }; static const struct ad4000_chip_info ad7988_1_chip_info = { .dev_name = "ad7988-1", .chan_spec = AD4000_PSEUDO_DIFF_CHANNELS('u', 16, 0), + .offload_chan_spec = AD4000_PSEUDO_DIFF_CHANNEL('u', 16, 0, 1), .time_spec = &ad7988_1_t_spec, + .max_rate_hz = 100 * KILO, }; static const struct ad4000_chip_info ad7988_5_chip_info = { .dev_name = "ad7988-5", .chan_spec = AD4000_PSEUDO_DIFF_CHANNELS('u', 16, 0), + .offload_chan_spec = AD4000_PSEUDO_DIFF_CHANNEL('u', 16, 0, 1), .time_spec = &ad7686_t_spec, + .max_rate_hz = 500 * KILO, +}; + +static const struct spi_offload_config ad4000_offload_config = { + .capability_flags = SPI_OFFLOAD_CAP_TRIGGER | + SPI_OFFLOAD_CAP_RX_STREAM_DMA, }; struct ad4000_state { @@ -397,6 +508,13 @@ struct ad4000_state { struct gpio_desc *cnv_gpio; struct spi_transfer xfers[2]; struct spi_message msg; + struct spi_transfer offload_xfers[2]; + struct spi_message offload_msg; + struct spi_offload *offload; + struct spi_offload_trigger *offload_trigger; + bool using_offload; + unsigned long offload_trigger_hz; + int max_rate_hz; struct mutex lock; /* Protect read modify write cycle */ int vref_mv; enum ad4000_sdi sdi_pin; @@ -411,8 +529,10 @@ struct ad4000_state { */ struct { union { - __be16 sample_buf16; - __be32 sample_buf32; + __be16 sample_buf16_be; + __be32 sample_buf32_be; + u16 sample_buf16; + u32 sample_buf32; } data; aligned_s64 timestamp; } scan __aligned(IIO_DMA_MINALIGN); @@ -487,6 +607,25 @@ static int ad4000_read_reg(struct ad4000_state *st, unsigned int *val) return ret; } +static int ad4000_set_sampling_freq(struct ad4000_state *st, int freq) +{ + struct spi_offload_trigger_config config = { + .type = SPI_OFFLOAD_TRIGGER_PERIODIC, + .periodic = { + .frequency_hz = freq, + }, + }; + int ret; + + ret = spi_offload_trigger_validate(st->offload_trigger, &config); + if (ret) + return ret; + + st->offload_trigger_hz = config.periodic.frequency_hz; + + return 0; +} + static int ad4000_convert_and_acquire(struct ad4000_state *st) { int ret; @@ -515,10 +654,17 @@ static int ad4000_single_conversion(struct iio_dev *indio_dev, if (ret < 0) return ret; - if (chan->scan_type.storagebits > 16) - sample = be32_to_cpu(st->scan.data.sample_buf32); - else - sample = be16_to_cpu(st->scan.data.sample_buf16); + if (chan->scan_type.endianness == IIO_BE) { + if (chan->scan_type.realbits > 16) + sample = be32_to_cpu(st->scan.data.sample_buf32_be); + else + sample = be16_to_cpu(st->scan.data.sample_buf16_be); + } else { + if (chan->scan_type.realbits > 16) + sample = st->scan.data.sample_buf32; + else + sample = st->scan.data.sample_buf16; + } sample >>= chan->scan_type.shift; @@ -554,6 +700,9 @@ static int ad4000_read_raw(struct iio_dev *indio_dev, if (st->span_comp) *val = mult_frac(st->vref_mv, 1, 10); + return IIO_VAL_INT; + case IIO_CHAN_INFO_SAMP_FREQ: + *val = st->offload_trigger_hz; return IIO_VAL_INT; default: return -EINVAL; @@ -620,6 +769,7 @@ static int ad4000_write_raw(struct iio_dev *indio_dev, struct iio_chan_spec const *chan, int val, int val2, long mask) { + struct ad4000_state *st = iio_priv(indio_dev); int ret; switch (mask) { @@ -629,6 +779,15 @@ static int ad4000_write_raw(struct iio_dev *indio_dev, ret = __ad4000_write_raw(indio_dev, chan, val2); iio_device_release_direct(indio_dev); return ret; + case IIO_CHAN_INFO_SAMP_FREQ: + if (val < 1 || val > st->max_rate_hz) + return -EINVAL; + + if (!iio_device_claim_direct(indio_dev)) + return -EBUSY; + ret = ad4000_set_sampling_freq(st, val); + iio_device_release_direct(indio_dev); + return ret; default: return -EINVAL; } @@ -659,10 +818,115 @@ static const struct iio_info ad4000_reg_access_info = { .write_raw_get_fmt = &ad4000_write_raw_get_fmt, }; +static const struct iio_info ad4000_offload_info = { + .read_raw = &ad4000_read_raw, + .write_raw = &ad4000_write_raw, + .write_raw_get_fmt = &ad4000_write_raw_get_fmt, +}; + static const struct iio_info ad4000_info = { .read_raw = &ad4000_read_raw, }; +static int ad4000_offload_buffer_postenable(struct iio_dev *indio_dev) +{ + struct ad4000_state *st = iio_priv(indio_dev); + struct spi_offload_trigger_config config = { + .type = SPI_OFFLOAD_TRIGGER_PERIODIC, + .periodic = { + .frequency_hz = st->offload_trigger_hz, + }, + }; + + return spi_offload_trigger_enable(st->offload, st->offload_trigger, + &config); +} + +static int ad4000_offload_buffer_predisable(struct iio_dev *indio_dev) +{ + struct ad4000_state *st = iio_priv(indio_dev); + + spi_offload_trigger_disable(st->offload, st->offload_trigger); + + return 0; +} + +static const struct iio_buffer_setup_ops ad4000_offload_buffer_setup_ops = { + .postenable = &ad4000_offload_buffer_postenable, + .predisable = &ad4000_offload_buffer_predisable, +}; + +static int ad4000_spi_offload_setup(struct iio_dev *indio_dev, + struct ad4000_state *st) +{ + struct spi_device *spi = st->spi; + struct device *dev = &spi->dev; + struct dma_chan *rx_dma; + int ret; + + st->offload_trigger = devm_spi_offload_trigger_get(dev, st->offload, + SPI_OFFLOAD_TRIGGER_PERIODIC); + if (IS_ERR(st->offload_trigger)) + return dev_err_probe(dev, PTR_ERR(st->offload_trigger), + "Failed to get offload trigger\n"); + + ret = ad4000_set_sampling_freq(st, st->max_rate_hz); + if (ret) + return dev_err_probe(dev, ret, + "Failed to set sampling frequency\n"); + + rx_dma = devm_spi_offload_rx_stream_request_dma_chan(dev, st->offload); + if (IS_ERR(rx_dma)) + return dev_err_probe(dev, PTR_ERR(rx_dma), + "Failed to get offload RX DMA\n"); + + ret = devm_iio_dmaengine_buffer_setup_with_handle(dev, indio_dev, rx_dma, + IIO_BUFFER_DIRECTION_IN); + if (ret) + return dev_err_probe(dev, ret, "Failed to setup DMA buffer\n"); + + return 0; +} + +/* + * This executes a data sample transfer when using SPI offloading for when the + * device connections are in "3-wire" mode, selected when the adi,sdi-pin device + * tree property is set to "high". In this connection mode, the ADC SDI pin is + * connected to VIO and ADC CNV pin is connected to a SPI controller CS (it + * can't be connected to a GPIO). + * + * In order to achieve the maximum sample rate, we only do one transfer per + * SPI offload trigger. Because the ADC output has a one sample latency (delay) + * when the device is wired in "3-wire" mode and only one transfer per sample is + * being made in turbo mode, the first data sample is not valid because it + * contains the output of an earlier conversion result. We also set transfer + * `bits_per_word` to achieve higher throughput by using the minimum number of + * SCLK cycles. Also, a delay is added to make sure we meet the minimum quiet + * time before releasing the CS line. + * + * Note that, with `bits_per_word` set to the number of ADC precision bits, + * transfers use larger word sizes that get stored in 'in-memory wordsizes' that + * are always in native CPU byte order. Because of that, IIO buffer elements + * ought to be read in CPU endianness which requires setting IIO scan_type + * endianness accordingly (i.e. IIO_CPU). + */ +static int ad4000_prepare_offload_message(struct ad4000_state *st, + const struct iio_chan_spec *chan) +{ + struct spi_transfer *xfers = st->offload_xfers; + + xfers[0].bits_per_word = chan->scan_type.realbits; + xfers[0].len = chan->scan_type.realbits > 16 ? 4 : 2; + xfers[0].delay.value = st->time_spec->t_quiet2_ns; + xfers[0].delay.unit = SPI_DELAY_UNIT_NSECS; + xfers[0].offload_flags = SPI_OFFLOAD_XFER_RX_STREAM; + + spi_message_init_with_transfers(&st->offload_msg, xfers, 1); + st->offload_msg.offload = st->offload; + + return devm_spi_optimize_message(&st->spi->dev, st->spi, &st->offload_msg); +} + /* * This executes a data sample transfer for when the device connections are * in "3-wire" mode, selected when the adi,sdi-pin device tree property is @@ -690,6 +954,15 @@ static int ad4000_prepare_3wire_mode_message(struct ad4000_state *st, xfers[1].rx_buf = &st->scan.data; xfers[1].len = BITS_TO_BYTES(chan->scan_type.storagebits); + + /* + * If the device is set up for SPI offloading, IIO channel scan_type is + * set to IIO_CPU. When that is the case, use larger SPI word sizes for + * single-shot reads too. Thus, sample data can be correctly handled in + * ad4000_single_conversion() according to scan_type endianness. + */ + if (chan->scan_type.endianness != IIO_BE) + xfers[1].bits_per_word = chan->scan_type.realbits; xfers[1].delay.value = st->time_spec->t_quiet2_ns; xfers[1].delay.unit = SPI_DELAY_UNIT_NSECS; @@ -733,6 +1006,9 @@ static int ad4000_config(struct ad4000_state *st) if (device_property_present(&st->spi->dev, "adi,high-z-input")) reg_val |= FIELD_PREP(AD4000_CFG_HIGHZ, 1); + if (st->using_offload) + reg_val |= FIELD_PREP(AD4000_CFG_TURBO, 1); + return ad4000_write_reg(st, reg_val); } @@ -755,6 +1031,7 @@ static int ad4000_probe(struct spi_device *spi) st = iio_priv(indio_dev); st->spi = spi; st->time_spec = chip->time_spec; + st->max_rate_hz = chip->max_rate_hz; ret = devm_regulator_bulk_get_enable(dev, ARRAY_SIZE(ad4000_power_supplies), ad4000_power_supplies); @@ -772,6 +1049,26 @@ static int ad4000_probe(struct spi_device *spi) return dev_err_probe(dev, PTR_ERR(st->cnv_gpio), "Failed to get CNV GPIO"); + st->offload = devm_spi_offload_get(dev, spi, &ad4000_offload_config); + ret = PTR_ERR_OR_ZERO(st->offload); + if (ret && ret != -ENODEV) + return dev_err_probe(dev, ret, "Failed to get offload\n"); + + st->using_offload = !IS_ERR(st->offload); + if (st->using_offload) { + indio_dev->setup_ops = &ad4000_offload_buffer_setup_ops; + ret = ad4000_spi_offload_setup(indio_dev, st); + if (ret) + return ret; + } else { + ret = devm_iio_triggered_buffer_setup(dev, indio_dev, + &iio_pollfunc_store_time, + &ad4000_trigger_handler, + NULL); + if (ret) + return ret; + } + ret = device_property_match_property_string(dev, "adi,sdi-pin", ad4000_sdi_pin, ARRAY_SIZE(ad4000_sdi_pin)); @@ -784,7 +1081,10 @@ static int ad4000_probe(struct spi_device *spi) switch (st->sdi_pin) { case AD4000_SDI_MOSI: indio_dev->info = &ad4000_reg_access_info; - indio_dev->channels = chip->reg_access_chan_spec; + + /* Set CNV/CS high time for when turbo mode is used */ + spi->cs_inactive.value = st->time_spec->t_quiet1_ns; + spi->cs_inactive.unit = SPI_DELAY_UNIT_NSECS; /* * In "3-wire mode", the ADC SDI line must be kept high when @@ -796,9 +1096,26 @@ static int ad4000_probe(struct spi_device *spi) if (ret < 0) return ret; + if (st->using_offload) { + indio_dev->channels = &chip->reg_access_offload_chan_spec; + indio_dev->num_channels = 1; + ret = ad4000_prepare_offload_message(st, indio_dev->channels); + if (ret) + return dev_err_probe(dev, ret, + "Failed to optimize SPI msg\n"); + } else { + indio_dev->channels = chip->reg_access_chan_spec; + indio_dev->num_channels = ARRAY_SIZE(chip->reg_access_chan_spec); + } + + /* + * Call ad4000_prepare_3wire_mode_message() so single-shot read + * SPI messages are always initialized. + */ ret = ad4000_prepare_3wire_mode_message(st, &indio_dev->channels[0]); if (ret) - return ret; + return dev_err_probe(dev, ret, + "Failed to optimize SPI msg\n"); ret = ad4000_config(st); if (ret < 0) @@ -806,19 +1123,47 @@ static int ad4000_probe(struct spi_device *spi) break; case AD4000_SDI_VIO: - indio_dev->info = &ad4000_info; - indio_dev->channels = chip->chan_spec; + if (st->using_offload) { + indio_dev->info = &ad4000_offload_info; + indio_dev->channels = &chip->offload_chan_spec; + indio_dev->num_channels = 1; + + /* Set CNV/CS high time for when turbo mode is not used */ + if (!st->cnv_gpio) { + spi->cs_inactive.value = st->time_spec->t_conv_ns; + spi->cs_inactive.unit = SPI_DELAY_UNIT_NSECS; + ret = spi_setup(spi); + if (ret < 0) + return ret; + } + + ret = ad4000_prepare_offload_message(st, indio_dev->channels); + if (ret) + return dev_err_probe(dev, ret, + "Failed to optimize SPI msg\n"); + } else { + indio_dev->info = &ad4000_info; + indio_dev->channels = chip->chan_spec; + indio_dev->num_channels = ARRAY_SIZE(chip->chan_spec); + } + ret = ad4000_prepare_3wire_mode_message(st, &indio_dev->channels[0]); if (ret) - return ret; + return dev_err_probe(dev, ret, + "Failed to optimize SPI msg\n"); break; case AD4000_SDI_CS: + if (st->using_offload) + return dev_err_probe(dev, -EPROTONOSUPPORT, + "Unsupported sdi-pin + offload config\n"); indio_dev->info = &ad4000_info; indio_dev->channels = chip->chan_spec; + indio_dev->num_channels = ARRAY_SIZE(chip->chan_spec); ret = ad4000_prepare_4wire_mode_message(st, &indio_dev->channels[0]); if (ret) - return ret; + return dev_err_probe(dev, ret, + "Failed to optimize SPI msg\n"); break; case AD4000_SDI_GND: @@ -830,7 +1175,6 @@ static int ad4000_probe(struct spi_device *spi) } indio_dev->name = chip->dev_name; - indio_dev->num_channels = 2; ret = devm_mutex_init(dev, &st->lock); if (ret) @@ -853,12 +1197,6 @@ static int ad4000_probe(struct spi_device *spi) ad4000_fill_scale_tbl(st, &indio_dev->channels[0]); - ret = devm_iio_triggered_buffer_setup(dev, indio_dev, - &iio_pollfunc_store_time, - &ad4000_trigger_handler, NULL); - if (ret) - return ret; - return devm_iio_device_register(dev, indio_dev); } @@ -947,3 +1285,4 @@ module_spi_driver(ad4000_driver); MODULE_AUTHOR("Marcelo Schmitt <marcelo.schmitt@analog.com>"); MODULE_DESCRIPTION("Analog Devices AD4000 ADC driver"); MODULE_LICENSE("GPL"); +MODULE_IMPORT_NS("IIO_DMAENGINE_BUFFER"); -- 2.47.2 ^ permalink raw reply related [flat|nested] 12+ messages in thread
* Re: [PATCH v3 1/4] iio: adc: ad4000: Add support for SPI offload 2025-03-26 13:24 ` [PATCH v3 1/4] iio: adc: ad4000: Add support for SPI offload Marcelo Schmitt @ 2025-03-27 16:12 ` David Lechner 2025-03-27 17:56 ` Marcelo Schmitt 0 siblings, 1 reply; 12+ messages in thread From: David Lechner @ 2025-03-27 16:12 UTC (permalink / raw) To: Marcelo Schmitt, linux-iio, linux-doc, linux-kernel Cc: jic23, lars, Michael.Hennerich, corbet, marcelo.schmitt1 On 3/26/25 8:24 AM, Marcelo Schmitt wrote: > FPGA HDL projects can include a PWM generator in addition to SPI-Engine. > The PWM IP is used to trigger SPI-Engine offload modules that in turn set > SPI-Engine to execute transfers to poll data from the ADC. That allows data > to be read at the maximum sample rates. Also, it is possible to set a > specific sample rate by setting the proper PWM duty cycle and related state > parameters, thus allowing an adjustable ADC sample rate when a PWM (offload > trigger) is used in combination with SPI-Engine. > > Add support for SPI offload. > > Signed-off-by: Marcelo Schmitt <marcelo.schmitt@analog.com> > --- ... > diff --git a/drivers/iio/adc/ad4000.c b/drivers/iio/adc/ad4000.c > index 4fe8dee48da9..9fc56853265e 100644 > --- a/drivers/iio/adc/ad4000.c > +++ b/drivers/iio/adc/ad4000.c > @@ -16,11 +16,13 @@ > #include <linux/gpio/consumer.h> > #include <linux/regulator/consumer.h> > #include <linux/spi/spi.h> > +#include <linux/spi/offload/consumer.h> Alphabetical order? > #include <linux/units.h> > #include <linux/util_macros.h> > #include <linux/iio/iio.h> > > #include <linux/iio/buffer.h> > +#include <linux/iio/buffer-dmaengine.h> > #include <linux/iio/triggered_buffer.h> > #include <linux/iio/trigger_consumer.h> > ... > > +/* > + * When SPI offload is configured, transfers are executed withouth CPU s/withouth/without/ > + * intervention so no soft timestamp can be recorded when transfers run. > + * Because of that, the macros that set timestamp channel are only used when > + * transfers are not offloaded. > + */ ... > @@ -784,7 +1081,10 @@ static int ad4000_probe(struct spi_device *spi) > switch (st->sdi_pin) { > case AD4000_SDI_MOSI: > indio_dev->info = &ad4000_reg_access_info; > - indio_dev->channels = chip->reg_access_chan_spec; > + > + /* Set CNV/CS high time for when turbo mode is used */ > + spi->cs_inactive.value = st->time_spec->t_quiet1_ns; > + spi->cs_inactive.unit = SPI_DELAY_UNIT_NSECS; This code path later calls ad4000_prepare_3wire_mode_message() which sets: xfers[0].cs_change_delay.value = st->time_spec->t_conv_ns; Which contradicts/overrides this. > > /* > * In "3-wire mode", the ADC SDI line must be kept high when > @@ -796,9 +1096,26 @@ static int ad4000_probe(struct spi_device *spi) > if (ret < 0) > return ret; > > + if (st->using_offload) { > + indio_dev->channels = &chip->reg_access_offload_chan_spec; > + indio_dev->num_channels = 1; > + ret = ad4000_prepare_offload_message(st, indio_dev->channels); > + if (ret) > + return dev_err_probe(dev, ret, > + "Failed to optimize SPI msg\n"); > + } else { > + indio_dev->channels = chip->reg_access_chan_spec; > + indio_dev->num_channels = ARRAY_SIZE(chip->reg_access_chan_spec); > + } > + > + /* > + * Call ad4000_prepare_3wire_mode_message() so single-shot read > + * SPI messages are always initialized. > + */ > ret = ad4000_prepare_3wire_mode_message(st, &indio_dev->channels[0]); > if (ret) > - return ret; > + return dev_err_probe(dev, ret, > + "Failed to optimize SPI msg\n"); > > ret = ad4000_config(st); > if (ret < 0) > @@ -806,19 +1123,47 @@ static int ad4000_probe(struct spi_device *spi) > > break; > case AD4000_SDI_VIO: > - indio_dev->info = &ad4000_info; > - indio_dev->channels = chip->chan_spec; > + if (st->using_offload) { > + indio_dev->info = &ad4000_offload_info; > + indio_dev->channels = &chip->offload_chan_spec; > + indio_dev->num_channels = 1; > + > + /* Set CNV/CS high time for when turbo mode is not used */ > + if (!st->cnv_gpio) { > + spi->cs_inactive.value = st->time_spec->t_conv_ns; > + spi->cs_inactive.unit = SPI_DELAY_UNIT_NSECS; I'm still not sold on this. We know it has no effect with AXI SPI Engine and it is writing over something that usually comes from DT. It is misleading. And the non-offload case already does: xfers[0].cs_change_delay.value = st->time_spec->t_conv_ns; which actually does work with the AXI SPI Engine. So why not be consistent and do it the same way for the offload case? It also seems safe to omit this altogether in the offload case and assume that the max sample rate will also ensure that the miniumum time for CS deasserted is respected. > + ret = spi_setup(spi); > + if (ret < 0) > + return ret; > + } > + > + ret = ad4000_prepare_offload_message(st, indio_dev->channels); > + if (ret) > + return dev_err_probe(dev, ret, > + "Failed to optimize SPI msg\n"); > + } else { > + indio_dev->info = &ad4000_info; > + indio_dev->channels = chip->chan_spec; > + indio_dev->num_channels = ARRAY_SIZE(chip->chan_spec); > + } > + > ret = ad4000_prepare_3wire_mode_message(st, &indio_dev->channels[0]); > if (ret) > - return ret; > + return dev_err_probe(dev, ret, > + "Failed to optimize SPI msg\n"); > > break; ^ permalink raw reply [flat|nested] 12+ messages in thread
* Re: [PATCH v3 1/4] iio: adc: ad4000: Add support for SPI offload 2025-03-27 16:12 ` David Lechner @ 2025-03-27 17:56 ` Marcelo Schmitt 2025-03-27 18:56 ` David Lechner 0 siblings, 1 reply; 12+ messages in thread From: Marcelo Schmitt @ 2025-03-27 17:56 UTC (permalink / raw) To: David Lechner Cc: Marcelo Schmitt, linux-iio, linux-doc, linux-kernel, jic23, lars, Michael.Hennerich, corbet Hi David, thank you for your review. Hope I don't sound harsh on my reply. Comments inline. Thanks On 03/27, David Lechner wrote: > On 3/26/25 8:24 AM, Marcelo Schmitt wrote: > > FPGA HDL projects can include a PWM generator in addition to SPI-Engine. > > The PWM IP is used to trigger SPI-Engine offload modules that in turn set > > SPI-Engine to execute transfers to poll data from the ADC. That allows data > > to be read at the maximum sample rates. Also, it is possible to set a > > specific sample rate by setting the proper PWM duty cycle and related state > > parameters, thus allowing an adjustable ADC sample rate when a PWM (offload > > trigger) is used in combination with SPI-Engine. > > > > Add support for SPI offload. > > > > Signed-off-by: Marcelo Schmitt <marcelo.schmitt@analog.com> > > --- > ... > > #include <linux/regulator/consumer.h> > > #include <linux/spi/spi.h> > > +#include <linux/spi/offload/consumer.h> > > Alphabetical order? Ah drat, yes. Hope I've put that in the correct order now. > > > #include <linux/units.h> > > #include <linux/util_macros.h> > > #include <linux/iio/iio.h> > > > > #include <linux/iio/buffer.h> > > +#include <linux/iio/buffer-dmaengine.h> > > #include <linux/iio/triggered_buffer.h> > > #include <linux/iio/trigger_consumer.h> > > Also changing to ... #include <linux/units.h> #include <linux/util_macros.h> #include <linux/iio/iio.h> #include <linux/iio/buffer.h> #include <linux/iio/buffer-dmaengine.h> ... > > ... > > > > > +/* > > + * When SPI offload is configured, transfers are executed withouth CPU > > s/withouth/without/ Ack > > + * intervention so no soft timestamp can be recorded when transfers run. > > + * Because of that, the macros that set timestamp channel are only used when > > + * transfers are not offloaded. > > + */ ... > > @@ -784,7 +1081,10 @@ static int ad4000_probe(struct spi_device *spi) > > switch (st->sdi_pin) { > > case AD4000_SDI_MOSI: > > indio_dev->info = &ad4000_reg_access_info; > > - indio_dev->channels = chip->reg_access_chan_spec; > > + > > + /* Set CNV/CS high time for when turbo mode is used */ > > + spi->cs_inactive.value = st->time_spec->t_quiet1_ns; > > + spi->cs_inactive.unit = SPI_DELAY_UNIT_NSECS; > > This code path later calls ad4000_prepare_3wire_mode_message() which sets: > > xfers[0].cs_change_delay.value = st->time_spec->t_conv_ns; > > Which contradicts/overrides this. Oof, good point. Though, it's probably not a problem if single-shot transfers use a longer delay, right? Would that also override cs_inactive or be of any trouble for SPI controllers? ... > > > > + if (st->using_offload) { > > + indio_dev->channels = &chip->reg_access_offload_chan_spec; > > + indio_dev->num_channels = 1; > > + ret = ad4000_prepare_offload_message(st, indio_dev->channels); > > + if (ret) > > + return dev_err_probe(dev, ret, > > + "Failed to optimize SPI msg\n"); > > + } else { > > + indio_dev->channels = chip->reg_access_chan_spec; > > + indio_dev->num_channels = ARRAY_SIZE(chip->reg_access_chan_spec); > > + } > > + > > + /* > > + * Call ad4000_prepare_3wire_mode_message() so single-shot read > > + * SPI messages are always initialized. > > + */ > > ret = ad4000_prepare_3wire_mode_message(st, &indio_dev->channels[0]); > > if (ret) > > - return ret; > > + return dev_err_probe(dev, ret, > > + "Failed to optimize SPI msg\n"); > > > > ret = ad4000_config(st); > > if (ret < 0) > > @@ -806,19 +1123,47 @@ static int ad4000_probe(struct spi_device *spi) > > > > break; > > case AD4000_SDI_VIO: > > - indio_dev->info = &ad4000_info; > > - indio_dev->channels = chip->chan_spec; > > + if (st->using_offload) { > > + indio_dev->info = &ad4000_offload_info; > > + indio_dev->channels = &chip->offload_chan_spec; > > + indio_dev->num_channels = 1; > > + > > + /* Set CNV/CS high time for when turbo mode is not used */ > > + if (!st->cnv_gpio) { > > + spi->cs_inactive.value = st->time_spec->t_conv_ns; > > + spi->cs_inactive.unit = SPI_DELAY_UNIT_NSECS; > > I'm still not sold on this. We know it has no effect with AXI SPI Engine and > it is writing over something that usually comes from DT. It is misleading. I thought it was okay to set cs_inactive and call spi_setup() from the field doc in include/linux/spi/spi.h. set_cs_timing() method is for SPI controllers that supports configuring CS timing. This hook allows SPI client drivers to request SPI controllers to configure specific CS timing through spi_set_cs_timing() after spi_setup(). Would it be better to set spi-cs-inactive-delay-ns in ADC dt node? Or it still doesn't look like a proper use of cs_inactive? > > And the non-offload case already does: > > xfers[0].cs_change_delay.value = st->time_spec->t_conv_ns; > > which actually does work with the AXI SPI Engine. So why not be consistent and > do it the same way for the offload case? One of the points in using `bits_per_word` in spi transfers was to reach high frequency sample rate, right? I think it makes sense to use them for SPI offload transfers. But we were also trying to set a proper CNV/CS dealy so that ADC conversion could complete properly before starting requesting the data. That also sound reasonable to me. But, spi_transfer struct doesn't provide a good way of setting a CS inactive delay if only one transfer is executed. If we use `cs_change_delay`, we would then be running two transfers, no? Plus, the ADC would be doing two conversions (one after CS deasert of previous message and one after CS deassert at the end of the first transfer) while we only read one of them. The offload message preparation would look like what we had in v2: static int ad4000_prepare_offload_turbo_message(struct ad4000_state *st, const struct iio_chan_spec *chan) { struct spi_transfer *xfers = st->offload_xfers; /* Dummy transfer to guarantee enough CS high time. */ xfers[0].cs_change = 1; xfers[0].cs_change_delay.value = st->time_spec->t_quiet1_ns; xfers[0].cs_change_delay.unit = SPI_DELAY_UNIT_NSECS; xfers[1].bits_per_word = chan->scan_type.realbits; xfers[1].len = chan->scan_type.realbits > 16 ? 4 : 2; xfers[1].delay.value = st->time_spec->t_quiet2_ns; xfers[1].delay.unit = SPI_DELAY_UNIT_NSECS; xfers[1].offload_flags = SPI_OFFLOAD_XFER_RX_STREAM; spi_message_init_with_transfers(&st->offload_msg, xfers, 2); st->offload_msg.offload = st->offload; return devm_spi_optimize_message(&st->spi->dev, st->spi, &st->offload_msg); } Are we worried about a few clock cycles in between transfers but not worried about running an entire dummy transfer? Plus, I've tested the single-transfer offload message version with ADAQ4003 on CoraZ7 and verified the results were correct. FWIW, I put a copy of the dts I used for the tests at the end of this email. > > It also seems safe to omit this altogether in the offload case and assume that > the max sample rate will also ensure that the miniumum time for CS deasserted > is respected. If we can assume that, then I think that's another reason why we don't need a dummy transfer to set CS high delay. > > > + ret = spi_setup(spi); > > + if (ret < 0) > > + return ret; > > + } > > + > > + ret = ad4000_prepare_offload_message(st, indio_dev->channels); > > + if (ret) > > + return dev_err_probe(dev, ret, > > + "Failed to optimize SPI msg\n"); > > + } else { > > + indio_dev->info = &ad4000_info; > > + indio_dev->channels = chip->chan_spec; > > + indio_dev->num_channels = ARRAY_SIZE(chip->chan_spec); > > + } > > + > > ret = ad4000_prepare_3wire_mode_message(st, &indio_dev->channels[0]); > > if (ret) > > - return ret; > > + return dev_err_probe(dev, ret, > > + "Failed to optimize SPI msg\n"); > > > > break; ---------- zynq-coraz7s-adaq4003.dts { ------------------ // SPDX-License-Identifier: GPL-2.0 /* * Analog Devices ADAQ4003 * * hdl_project: <pulsar_adc/coraz7s> * Link: https://github.com/analogdevicesinc/hdl/tree/main/projects/pulsar_adc * board_revision: <A> * * Copyright (C) 2023 Analog Devices Inc. */ /dts-v1/; #include "zynq-coraz7s.dtsi" #include <dt-bindings/interrupt-controller/irq.h> #include <dt-bindings/gpio/gpio.h> / { adc_vref: regulator-vref { compatible = "regulator-fixed"; regulator-name = "EVAL 5V Vref"; regulator-min-microvolt = <5000000>; regulator-max-microvolt = <5000000>; regulator-always-on; }; adc_vdd: regulator-vdd { compatible = "regulator-fixed"; regulator-name = "Eval VDD supply"; regulator-min-microvolt = <5000000>; regulator-max-microvolt = <5000000>; regulator-always-on; }; adc_vio: regulator-vio { compatible = "regulator-fixed"; regulator-name = "Eval VIO supply"; regulator-min-microvolt = <3300000>; regulator-max-microvolt = <3300000>; regulator-always-on; }; trigger_pwm: adc-pwm-trigger { compatible = "pwm-trigger"; #trigger-source-cells = <0>; pwms = <&adc_trigger 0 1000000 0>; }; }; &fpga_axi { adc_trigger: pwm@0x44b00000 { compatible = "adi,axi-pwmgen-2.00.a"; reg = <0x44b00000 0x1000>; label = "adc_conversion_trigger"; #pwm-cells = <2>; clocks = <&spi_clk>; }; spi_engine: spi@44a00000 { compatible = "adi,axi-spi-engine-1.00.a"; reg = <0x44a00000 0x10000>; interrupt-parent = <&intc>; interrupts = <0 56 IRQ_TYPE_LEVEL_HIGH>; clocks = <&clkc 15 &spi_clk>; clock-names = "s_axi_aclk", "spi_clk"; dmas = <&rx_dma 0>; dma-names = "offload0-rx"; trigger-sources = <&trigger_pwm>; #address-cells = <0x1>; #size-cells = <0x0>; adc@0 { compatible = "adi,adaq4003"; reg = <0>; spi-max-frequency = <100000000>; vdd-supply = <&adc_vdd>; vio-supply = <&adc_vio>; ref-supply = <&adc_vref>; adi,high-z-input; adi,gain-milli = /bits/ 16 <454>; }; }; rx_dma: rx-dmac@44a30000 { compatible = "adi,axi-dmac-1.00.a"; reg = <0x44a30000 0x1000>; #dma-cells = <1>; interrupt-parent = <&intc>; interrupts = <0 57 IRQ_TYPE_LEVEL_HIGH>; clocks = <&clkc 16>; }; spi_clk: axi-clkgen@0x44a70000 { compatible = "adi,axi-clkgen-2.00.a"; reg = <0x44a70000 0x10000>; #clock-cells = <0>; clocks = <&clkc 15>, <&clkc 15>; clock-names = "s_axi_aclk", "clkin1"; clock-output-names = "spi_clk"; assigned-clocks = <&spi_clk>; assigned-clock-rates = <200000000>; }; }; ---------- } zynq-coraz7s-adaq4003.dts ------------------ ^ permalink raw reply [flat|nested] 12+ messages in thread
* Re: [PATCH v3 1/4] iio: adc: ad4000: Add support for SPI offload 2025-03-27 17:56 ` Marcelo Schmitt @ 2025-03-27 18:56 ` David Lechner 0 siblings, 0 replies; 12+ messages in thread From: David Lechner @ 2025-03-27 18:56 UTC (permalink / raw) To: Marcelo Schmitt Cc: Marcelo Schmitt, linux-iio, linux-doc, linux-kernel, jic23, lars, Michael.Hennerich, corbet On 3/27/25 12:56 PM, Marcelo Schmitt wrote: ... >>> case AD4000_SDI_VIO: >>> - indio_dev->info = &ad4000_info; >>> - indio_dev->channels = chip->chan_spec; >>> + if (st->using_offload) { >>> + indio_dev->info = &ad4000_offload_info; >>> + indio_dev->channels = &chip->offload_chan_spec; >>> + indio_dev->num_channels = 1; >>> + >>> + /* Set CNV/CS high time for when turbo mode is not used */ >>> + if (!st->cnv_gpio) { >>> + spi->cs_inactive.value = st->time_spec->t_conv_ns; >>> + spi->cs_inactive.unit = SPI_DELAY_UNIT_NSECS; >> >> I'm still not sold on this. We know it has no effect with AXI SPI Engine and >> it is writing over something that usually comes from DT. It is misleading. > > I thought it was okay to set cs_inactive and call spi_setup() from the field > doc in include/linux/spi/spi.h. > > set_cs_timing() method is for SPI controllers that supports > configuring CS timing. > > This hook allows SPI client drivers to request SPI controllers > to configure specific CS timing through spi_set_cs_timing() after > spi_setup(). > > Would it be better to set spi-cs-inactive-delay-ns in ADC dt node? > Or it still doesn't look like a proper use of cs_inactive? > >> >> And the non-offload case already does: >> >> xfers[0].cs_change_delay.value = st->time_spec->t_conv_ns; >> >> which actually does work with the AXI SPI Engine. So why not be consistent and >> do it the same way for the offload case? > > One of the points in using `bits_per_word` in spi transfers was to reach high > frequency sample rate, right? I think it makes sense to use them for SPI offload > transfers. But we were also trying to set a proper CNV/CS dealy so that ADC > conversion could complete properly before starting requesting the data. That > also sound reasonable to me. But, spi_transfer struct doesn't provide a good > way of setting a CS inactive delay if only one transfer is executed. If we > use `cs_change_delay`, we would then be running two transfers, no? We would still only be doing one xfer per SPI offload trigger. The only difference it would make is that it would ensure that the second trigger could not come too soon after the CS deassert of the previous message. In other words, the sampling frequency is already supplying this delay between subsequent SPI messages. Setting xfers[0].cs_change_delay just adds insurance that if the next trigger comes too soon, it will be ignored. This could happen, e.g. if someone sets the max SCLK rate to something low enough that the single xfer takes longer than the trigger period at the max sampling frequency. In most cases though, we won't actually see cs_change_delay having any effect because the trigger to start the next SPI message doesn't come until later. > Plus, the ADC > would be doing two conversions (one after CS deasert of previous message and > one after CS deassert at the end of the first transfer) while we only read one > of them. > > The offload message preparation would look like what we had in v2: > > static int ad4000_prepare_offload_turbo_message(struct ad4000_state *st, > const struct iio_chan_spec *chan) > { > struct spi_transfer *xfers = st->offload_xfers; > > /* Dummy transfer to guarantee enough CS high time. */ > xfers[0].cs_change = 1; > xfers[0].cs_change_delay.value = st->time_spec->t_quiet1_ns; > xfers[0].cs_change_delay.unit = SPI_DELAY_UNIT_NSECS; > > xfers[1].bits_per_word = chan->scan_type.realbits; > xfers[1].len = chan->scan_type.realbits > 16 ? 4 : 2; > xfers[1].delay.value = st->time_spec->t_quiet2_ns; > xfers[1].delay.unit = SPI_DELAY_UNIT_NSECS; > xfers[1].offload_flags = SPI_OFFLOAD_XFER_RX_STREAM; > > spi_message_init_with_transfers(&st->offload_msg, xfers, 2); > st->offload_msg.offload = st->offload; > > return devm_spi_optimize_message(&st->spi->dev, st->spi, &st->offload_msg); > } > > Are we worried about a few clock cycles in between transfers but not worried > about running an entire dummy transfer? > > Plus, I've tested the single-transfer offload message version with ADAQ4003 on > CoraZ7 and verified the results were correct. > FWIW, I put a copy of the dts I used for the tests at the end of this email. > >> >> It also seems safe to omit this altogether in the offload case and assume that >> the max sample rate will also ensure that the miniumum time for CS deasserted >> is respected. > > If we can assume that, then I think that's another reason why we don't need > a dummy transfer to set CS high delay. > I agree we don't need two xfers, especially in turbo mode. ^ permalink raw reply [flat|nested] 12+ messages in thread
* [PATCH v3 2/4] Documentation: iio: ad4000: Add new supported parts 2025-03-26 13:24 [PATCH v3 0/4] iio: adc: ad4000: Add SPI offload support Marcelo Schmitt 2025-03-26 13:24 ` [PATCH v3 1/4] iio: adc: ad4000: Add support for SPI offload Marcelo Schmitt @ 2025-03-26 13:25 ` Marcelo Schmitt 2025-03-27 16:14 ` David Lechner 2025-03-26 13:25 ` [PATCH v3 3/4] Documentation: iio: ad4000: Add IIO Device characteristics section Marcelo Schmitt 2025-03-26 13:25 ` [PATCH v3 4/4] Documentation: iio: ad4000: Describe offload support Marcelo Schmitt 3 siblings, 1 reply; 12+ messages in thread From: Marcelo Schmitt @ 2025-03-26 13:25 UTC (permalink / raw) To: linux-iio, linux-doc, linux-kernel Cc: jic23, lars, Michael.Hennerich, corbet, dlechner, marcelo.schmitt1 Commit <c3948d090080> ("iio: adc: ad4000: Add support for PulSAR devices"), extended the ad4000 driver supports many single-channel PulSAR devices. Update IIO ad4000 documentation with the extra list of supported devices. Signed-off-by: Marcelo Schmitt <marcelo.schmitt@analog.com> --- Documentation/iio/ad4000.rst | 17 ++++++++++++++++- 1 file changed, 16 insertions(+), 1 deletion(-) diff --git a/Documentation/iio/ad4000.rst b/Documentation/iio/ad4000.rst index de8fd3ae6e62..5578a9cfd9d5 100644 --- a/Documentation/iio/ad4000.rst +++ b/Documentation/iio/ad4000.rst @@ -4,7 +4,7 @@ AD4000 driver ============= -Device driver for Analog Devices Inc. AD4000 series of ADCs. +Device driver for Analog Devices Inc. AD4000 series of ADCs and similar devices. Supported devices ================= @@ -25,6 +25,21 @@ Supported devices * `AD4022 <https://www.analog.com/AD4022>`_ * `ADAQ4001 <https://www.analog.com/ADAQ4001>`_ * `ADAQ4003 <https://www.analog.com/ADAQ4003>`_ +* `AD7685 <https://www.analog.com/AD7685>`_ +* `AD7686 <https://www.analog.com/AD7686>`_ +* `AD7687 <https://www.analog.com/AD7687>`_ +* `AD7688 <https://www.analog.com/AD7688>`_ +* `AD7690 <https://www.analog.com/AD7690>`_ +* `AD7691 <https://www.analog.com/AD7691>`_ +* `AD7693 <https://www.analog.com/AD7693>`_ +* `AD7942 <https://www.analog.com/AD7942>`_ +* `AD7946 <https://www.analog.com/AD7946>`_ +* `AD7980 <https://www.analog.com/AD7980>`_ +* `AD7982 <https://www.analog.com/AD7982>`_ +* `AD7983 <https://www.analog.com/AD7983>`_ +* `AD7984 <https://www.analog.com/AD7984>`_ +* `AD7988-1 <https://www.analog.com/AD7988-1>`_ +* `AD7988-5 <https://www.analog.com/AD7988-5>`_ Wiring connections ------------------ -- 2.47.2 ^ permalink raw reply related [flat|nested] 12+ messages in thread
* Re: [PATCH v3 2/4] Documentation: iio: ad4000: Add new supported parts 2025-03-26 13:25 ` [PATCH v3 2/4] Documentation: iio: ad4000: Add new supported parts Marcelo Schmitt @ 2025-03-27 16:14 ` David Lechner 2025-03-27 16:44 ` Marcelo Schmitt 0 siblings, 1 reply; 12+ messages in thread From: David Lechner @ 2025-03-27 16:14 UTC (permalink / raw) To: Marcelo Schmitt, linux-iio, linux-doc, linux-kernel Cc: jic23, lars, Michael.Hennerich, corbet, marcelo.schmitt1 On 3/26/25 8:25 AM, Marcelo Schmitt wrote: > Commit <c3948d090080> ("iio: adc: ad4000: Add support for PulSAR devices"), > extended the ad4000 driver supports many single-channel PulSAR devices. > > Update IIO ad4000 documentation with the extra list of supported devices. > > Signed-off-by: Marcelo Schmitt <marcelo.schmitt@analog.com> > --- Don't forget to pick up tags. I already gave my: Reviewed-by: David Lechner <dlechner@baylibre.com> ^ permalink raw reply [flat|nested] 12+ messages in thread
* Re: [PATCH v3 2/4] Documentation: iio: ad4000: Add new supported parts 2025-03-27 16:14 ` David Lechner @ 2025-03-27 16:44 ` Marcelo Schmitt 0 siblings, 0 replies; 12+ messages in thread From: Marcelo Schmitt @ 2025-03-27 16:44 UTC (permalink / raw) To: David Lechner Cc: Marcelo Schmitt, linux-iio, linux-doc, linux-kernel, jic23, lars, Michael.Hennerich, corbet On 03/27, David Lechner wrote: > On 3/26/25 8:25 AM, Marcelo Schmitt wrote: > > Commit <c3948d090080> ("iio: adc: ad4000: Add support for PulSAR devices"), > > extended the ad4000 driver supports many single-channel PulSAR devices. > > > > Update IIO ad4000 documentation with the extra list of supported devices. > > > > Signed-off-by: Marcelo Schmitt <marcelo.schmitt@analog.com> > > --- > Don't forget to pick up tags. I already gave my: Oops, sorry. My bad. I've picked them now. > > > Reviewed-by: David Lechner <dlechner@baylibre.com> > > Thanks, Marcelo ^ permalink raw reply [flat|nested] 12+ messages in thread
* [PATCH v3 3/4] Documentation: iio: ad4000: Add IIO Device characteristics section 2025-03-26 13:24 [PATCH v3 0/4] iio: adc: ad4000: Add SPI offload support Marcelo Schmitt 2025-03-26 13:24 ` [PATCH v3 1/4] iio: adc: ad4000: Add support for SPI offload Marcelo Schmitt 2025-03-26 13:25 ` [PATCH v3 2/4] Documentation: iio: ad4000: Add new supported parts Marcelo Schmitt @ 2025-03-26 13:25 ` Marcelo Schmitt 2025-03-27 16:14 ` David Lechner 2025-03-26 13:25 ` [PATCH v3 4/4] Documentation: iio: ad4000: Describe offload support Marcelo Schmitt 3 siblings, 1 reply; 12+ messages in thread From: Marcelo Schmitt @ 2025-03-26 13:25 UTC (permalink / raw) To: linux-iio, linux-doc, linux-kernel Cc: jic23, lars, Michael.Hennerich, corbet, dlechner, marcelo.schmitt1 Complement ad4000 IIO driver documentation with considerations about ``_scale_available`` attribute and table of typical channel attributes. Signed-off-by: Marcelo Schmitt <marcelo.schmitt@analog.com> --- Documentation/iio/ad4000.rst | 47 ++++++++++++++++++++++++++++++++++++ 1 file changed, 47 insertions(+) diff --git a/Documentation/iio/ad4000.rst b/Documentation/iio/ad4000.rst index 5578a9cfd9d5..468d30dc9214 100644 --- a/Documentation/iio/ad4000.rst +++ b/Documentation/iio/ad4000.rst @@ -144,3 +144,50 @@ Set ``adi,sdi-pin`` to ``"cs"`` to select this mode. ^ | | +--------------------| SCLK | +-------------+ + +IIO Device characteristics +========================== + +The AD4000 series driver supports differential and pseudo-differential ADCs. + +The span compression feature available in AD4000 series devices can be +enabled/disabled by changing the ``_scale_available`` attribute of the voltage +channel. Note that span compression configuration requires writing to AD4000 +configuration register, which is only possible when the ADC is wired in 3-wire +turbo mode, and the SPI controller is ``SPI_MOSI_IDLE_HIGH`` capable. If those +conditions are not met, no ``_scale_available`` attribute is provided. + +Besides that, differential and pseudo-differential voltage channels present +slightly different sysfs interfaces. + +Pseudo-differential ADCs +------------------------ + +Typical voltage channel attributes of a pseudo-differential AD4000 series device: + ++-------------------------------------------+------------------------------------------+ +| Voltage Channel Attributes | Description | ++===========================================+==========================================+ +| ``in_voltage0_raw`` | Raw ADC output code. | ++-------------------------------------------+------------------------------------------+ +| ``in_voltage0_offset`` | Offset to convert raw value to mV. | ++-------------------------------------------+------------------------------------------+ +| ``in_voltage0_scale`` | Scale factor to convert raw value to mV. | ++-------------------------------------------+------------------------------------------+ +| ``in_voltage0_scale_available`` | Toggles input span compression | ++-------------------------------------------+------------------------------------------+ + +Differential ADCs +----------------- + +Typical voltage channel attributes of a differential AD4000 series device: + ++-------------------------------------------+------------------------------------------+ +| Voltage Channel Attributes | Description | ++===========================================+==========================================+ +| ``in_voltage0-voltage1_raw`` | Raw ADC output code. | ++-------------------------------------------+------------------------------------------+ +| ``in_voltage0-voltage1_scale`` | Scale factor to convert raw value to mV. | ++-------------------------------------------+------------------------------------------+ +| ``in_voltage0-voltage1_scale_available`` | Toggles input span compression | ++-------------------------------------------+------------------------------------------+ -- 2.47.2 ^ permalink raw reply related [flat|nested] 12+ messages in thread
* Re: [PATCH v3 3/4] Documentation: iio: ad4000: Add IIO Device characteristics section 2025-03-26 13:25 ` [PATCH v3 3/4] Documentation: iio: ad4000: Add IIO Device characteristics section Marcelo Schmitt @ 2025-03-27 16:14 ` David Lechner 0 siblings, 0 replies; 12+ messages in thread From: David Lechner @ 2025-03-27 16:14 UTC (permalink / raw) To: Marcelo Schmitt, linux-iio, linux-doc, linux-kernel Cc: jic23, lars, Michael.Hennerich, corbet, marcelo.schmitt1 On 3/26/25 8:25 AM, Marcelo Schmitt wrote: > Complement ad4000 IIO driver documentation with considerations about > ``_scale_available`` attribute and table of typical channel attributes. > > Signed-off-by: Marcelo Schmitt <marcelo.schmitt@analog.com> > --- Reviewed-by: David Lechner <dlechner@baylibre.com> ^ permalink raw reply [flat|nested] 12+ messages in thread
* [PATCH v3 4/4] Documentation: iio: ad4000: Describe offload support 2025-03-26 13:24 [PATCH v3 0/4] iio: adc: ad4000: Add SPI offload support Marcelo Schmitt ` (2 preceding siblings ...) 2025-03-26 13:25 ` [PATCH v3 3/4] Documentation: iio: ad4000: Add IIO Device characteristics section Marcelo Schmitt @ 2025-03-26 13:25 ` Marcelo Schmitt 2025-03-27 16:15 ` David Lechner 3 siblings, 1 reply; 12+ messages in thread From: Marcelo Schmitt @ 2025-03-26 13:25 UTC (permalink / raw) To: linux-iio, linux-doc, linux-kernel Cc: jic23, lars, Michael.Hennerich, corbet, dlechner, marcelo.schmitt1 When SPI offloading is supported, the IIO device provides different sysfs interfaces to allow using the adjusting the sample rate. Document SPI offload support for AD4000 and similar devices. Signed-off-by: Marcelo Schmitt <marcelo.schmitt@analog.com> --- Documentation/iio/ad4000.rst | 22 ++++++++++++++++++++++ 1 file changed, 22 insertions(+) diff --git a/Documentation/iio/ad4000.rst b/Documentation/iio/ad4000.rst index 468d30dc9214..e490f9604b94 100644 --- a/Documentation/iio/ad4000.rst +++ b/Documentation/iio/ad4000.rst @@ -191,3 +191,25 @@ Typical voltage channel attributes of a differential AD4000 series device: +-------------------------------------------+------------------------------------------+ | ``in_voltage0-voltage1_scale_available`` | Toggles input span compression | +-------------------------------------------+------------------------------------------+ + +SPI offload support +------------------- + +To be able to achieve the maximum sample rate, the driver can be used with the +`AXI SPI Engine`_ to provide SPI offload support. + +.. _AXI SPI Engine: http://analogdevicesinc.github.io/hdl/projects/pulsar_adc/index.html + +When set for SPI offload support, the IIO device will provide different +interfaces. + +* Either ``in_voltage0_sampling_frequency`` or + ``in_voltage0-voltage1_sampling_frequency`` file is provided to allow setting + the sample rate. +* IIO trigger device is not provided (no ``trigger`` directory). +* ``timestamp`` channel is not provided. + +Also, because the ADC output has a one sample latency (delay) when the device is +wired in "3-wire" mode and only one transfer per sample is done when using SPI +offloading, the first data sample in the buffer is not valid because it contains +the output of an earlier conversion result. -- 2.47.2 ^ permalink raw reply related [flat|nested] 12+ messages in thread
* Re: [PATCH v3 4/4] Documentation: iio: ad4000: Describe offload support 2025-03-26 13:25 ` [PATCH v3 4/4] Documentation: iio: ad4000: Describe offload support Marcelo Schmitt @ 2025-03-27 16:15 ` David Lechner 0 siblings, 0 replies; 12+ messages in thread From: David Lechner @ 2025-03-27 16:15 UTC (permalink / raw) To: Marcelo Schmitt, linux-iio, linux-doc, linux-kernel Cc: jic23, lars, Michael.Hennerich, corbet, marcelo.schmitt1 On 3/26/25 8:25 AM, Marcelo Schmitt wrote: > When SPI offloading is supported, the IIO device provides different sysfs > interfaces to allow using the adjusting the sample rate. Document SPI > offload support for AD4000 and similar devices. > > Signed-off-by: Marcelo Schmitt <marcelo.schmitt@analog.com> > --- Reviewed-by: David Lechner <dlechner@baylibre.com> ^ permalink raw reply [flat|nested] 12+ messages in thread
end of thread, other threads:[~2025-03-27 18:56 UTC | newest] Thread overview: 12+ messages (download: mbox.gz follow: Atom feed -- links below jump to the message on this page -- 2025-03-26 13:24 [PATCH v3 0/4] iio: adc: ad4000: Add SPI offload support Marcelo Schmitt 2025-03-26 13:24 ` [PATCH v3 1/4] iio: adc: ad4000: Add support for SPI offload Marcelo Schmitt 2025-03-27 16:12 ` David Lechner 2025-03-27 17:56 ` Marcelo Schmitt 2025-03-27 18:56 ` David Lechner 2025-03-26 13:25 ` [PATCH v3 2/4] Documentation: iio: ad4000: Add new supported parts Marcelo Schmitt 2025-03-27 16:14 ` David Lechner 2025-03-27 16:44 ` Marcelo Schmitt 2025-03-26 13:25 ` [PATCH v3 3/4] Documentation: iio: ad4000: Add IIO Device characteristics section Marcelo Schmitt 2025-03-27 16:14 ` David Lechner 2025-03-26 13:25 ` [PATCH v3 4/4] Documentation: iio: ad4000: Describe offload support Marcelo Schmitt 2025-03-27 16:15 ` David Lechner
This is a public inbox, see mirroring instructions for how to clone and mirror all data and code used for this inbox; as well as URLs for NNTP newsgroup(s).