Message ID | 20240905082404.119022-9-aardelean@baylibre.com (mailing list archive) |
---|---|
State | Changes Requested |
Headers | show |
Series | iio: adc: ad7606: add support for AD7606C-{16,18} parts | expand |
On 9/5/24 3:24 AM, Alexandru Ardelean wrote: > The AD7606C-16 and AD7606C-18 are pretty similar with the AD7606B. > The main difference between AD7606C-16 & AD7606C-18 is the precision in > bits (16 vs 18). > Because of that, some scales need to be defined for the 18-bit variants, as > they need to be computed against 2**18 (vs 2**16 for the 16 bit-variants). > > Because the AD7606C-16,18 also supports bipolar & differential channels, > for SW-mode, the default range of 10 V or ±10V should be set at probe. > On reset, the default range (in the registers) is set to value 0x3 which > corresponds to '±10 V single-ended range', regardless of bipolar or > differential configuration. > > Aside from the scale/ranges, the AD7606C-16 is similar to the AD7606B. > > The AD7606C-18 variant offers 18-bit precision. Because of this, the > requirement to use this chip is that the SPI controller supports padding > of 18-bit sequences to 32-bit arrays. > > Datasheet links: > https://www.analog.com/media/en/technical-documentation/data-sheets/ad7606c-16.pdf > https://www.analog.com/media/en/technical-documentation/data-sheets/ad7606c-18.pdf > > Signed-off-by: Alexandru Ardelean <aardelean@baylibre.com> > --- > drivers/iio/adc/ad7606.c | 266 +++++++++++++++++++++++++++++++---- > drivers/iio/adc/ad7606.h | 17 ++- > drivers/iio/adc/ad7606_spi.c | 55 ++++++++ > 3 files changed, 309 insertions(+), 29 deletions(-) > > diff --git a/drivers/iio/adc/ad7606.c b/drivers/iio/adc/ad7606.c > index 4c3fbb28f790..999c4411859e 100644 > --- a/drivers/iio/adc/ad7606.c > +++ b/drivers/iio/adc/ad7606.c > @@ -28,14 +28,44 @@ > > #include "ad7606.h" > > +typedef void (*ad7606c_chan_setup_cb_t)(struct ad7606_state *st, int ch, > + bool bipolar, bool differential); > + > /* > * Scales are computed as 5000/32768 and 10000/32768 respectively, > * so that when applied to the raw values they provide mV values > */ > -static const unsigned int ad7606_scale_avail[2] = { > +static const unsigned int ad7606_16bit_hw_scale_avail[2] = { > 152588, 305176 > }; > > +static const unsigned int ad7606_18bit_hw_scale_avail[2] = { > + 38147, 76294 > +}; > + > +static const unsigned int ad7606c_16_scale_single_ended_unipolar_avail[3] = { > + 76294, 152588, 190735, > +}; > + > +static const unsigned int ad7606c_16_scale_single_ended_bipolar_avail[5] = { > + 76294, 152588, 190735, 305176, 381470 > +}; > + > +static const unsigned int ad7606c_16_scale_differential_bipolar_avail[4] = { > + 152588, 305176, 381470, 610352 > +}; > + > +static const unsigned int ad7606c_18_scale_single_ended_unipolar_avail[3] = { > + 19073, 38147, 47684 > +}; > + > +static const unsigned int ad7606c_18_scale_single_ended_bipolar_avail[5] = { > + 19073, 38147, 47684, 76294, 95367 > +}; > + > +static const unsigned int ad7606c_18_scale_differential_bipolar_avail[4] = { > + 38147, 76294, 95367, 152588 > +}; > > static const unsigned int ad7616_sw_scale_avail[3] = { > 76293, 152588, 305176 > @@ -82,11 +112,19 @@ static int ad7606_reg_access(struct iio_dev *indio_dev, > } > } > > -static int ad7606_read_samples(struct ad7606_state *st) > +static int ad7606_read_samples(struct ad7606_state *st, bool sign_extend_samples) > { > + unsigned int storagebits = st->chip_info->channels[1].scan_type.storagebits; Why [1]? Sure, they are all the same, but [0] would seem less arbitrary. > unsigned int num = st->chip_info->num_channels - 1; > - u16 *data = st->data; > - int ret; > + u32 *data32 = st->data.d32; > + u16 *data16 = st->data.d16; > + void *data; > + int i, ret; > + > + if (storagebits > 16) > + data = data32; > + else > + data = data16; > > /* > * The frstdata signal is set to high while and after reading the sample > @@ -108,11 +146,25 @@ static int ad7606_read_samples(struct ad7606_state *st) > return -EIO; > } > > - data++; > + if (storagebits > 16) > + data32++; > + else > + data16++; > num--; > } > > - return st->bops->read_block(st->dev, num, data); > + ret = st->bops->read_block(st->dev, num, data); Since data++ was removed, this looks broken now as well as the other read_block() not visible in the diff. Maybe better to drop data32 and data16, keep the change of data to void*, and change data++ to data += BITS_TO_BYTES(storagebits)? Although, all of this might be moot since it looks like this needs to be rebased on [1]. [1]: https://git.kernel.org/pub/scm/linux/kernel/git/jic23/iio.git/commit/?h=fixes-togreg&id=90826e08468ba7fb35d8b39645b22d9e80004afe > + if (ret) > + return ret; > + > + if (storagebits == 16 || !sign_extend_samples) > + return 0; > + > + /* For 18 bit samples, we need to sign-extend samples to 32 bits */ > + for (i = 0; i < num; i++) > + data32[i] = sign_extend32(data32[i], 17);> + > + return 0; > } > > static irqreturn_t ad7606_trigger_handler(int irq, void *p) > @@ -124,11 +176,11 @@ static irqreturn_t ad7606_trigger_handler(int irq, void *p) > > guard(mutex)(&st->lock); > > - ret = ad7606_read_samples(st); > + ret = ad7606_read_samples(st, true); Shouldn't the sign_extend parameter depend on if the data is unipolar or bipolar? > if (ret) > goto error_ret; > > - iio_push_to_buffers_with_timestamp(indio_dev, st->data, > + iio_push_to_buffers_with_timestamp(indio_dev, st->data.d16, > iio_get_time_ns(indio_dev)); > error_ret: > iio_trigger_notify_done(indio_dev->trig); > @@ -142,6 +194,7 @@ static int ad7606_scan_direct(struct iio_dev *indio_dev, unsigned int ch, > int *val) > { > struct ad7606_state *st = iio_priv(indio_dev); > + unsigned int storagebits = st->chip_info->channels[1].scan_type.storagebits; > int ret; > > gpiod_set_value(st->gpio_convst, 1); > @@ -152,9 +205,13 @@ static int ad7606_scan_direct(struct iio_dev *indio_dev, unsigned int ch, > goto error_ret; > } > > - ret = ad7606_read_samples(st); > - if (ret == 0) > - *val = sign_extend32(st->data[ch], 15); > + ret = ad7606_read_samples(st, false); Why not let ad7606_read_samples() do the sign extending since it can do that now? > + if (ret == 0) { > + if (storagebits > 16) > + *val = sign_extend32(st->data.d32[ch], 17); > + else > + *val = sign_extend32(st->data.d16[ch], 15); > + } > > error_ret: > gpiod_set_value(st->gpio_convst, 0); > @@ -267,7 +324,7 @@ static int ad7606_write_raw(struct iio_dev *indio_dev, > ch = chan->address; > cs = &st->chan_scales[ch]; > i = find_closest(val2, cs->scale_avail, cs->num_scales); > - ret = st->write_scale(indio_dev, ch, i); > + ret = st->write_scale(indio_dev, ch, i + cs->reg_offset); > if (ret < 0) > return ret; > cs->range = i; > @@ -350,6 +407,18 @@ static const struct iio_chan_spec ad7606_channels_16bit[] = { > AD7606_CHANNEL(7, 16), > }; > > +static const struct iio_chan_spec ad7606_channels_18bit[] = { > + IIO_CHAN_SOFT_TIMESTAMP(8), > + AD7606_CHANNEL(0, 18), > + AD7606_CHANNEL(1, 18), > + AD7606_CHANNEL(2, 18), > + AD7606_CHANNEL(3, 18), > + AD7606_CHANNEL(4, 18), > + AD7606_CHANNEL(5, 18), > + AD7606_CHANNEL(6, 18), > + AD7606_CHANNEL(7, 18), > +}; > + > /* > * The current assumption that this driver makes for AD7616, is that it's > * working in Hardware Mode with Serial, Burst and Sequencer modes activated. > @@ -410,6 +479,18 @@ static const struct ad7606_chip_info ad7606_chip_info_tbl[] = { > .oversampling_avail = ad7606_oversampling_avail, > .oversampling_num = ARRAY_SIZE(ad7606_oversampling_avail), > }, > + [ID_AD7606C_16] = { > + .channels = ad7606_channels_16bit, > + .num_channels = 9, Could be nice to have a cleanup patch before this to convert others to use ARRAY_SIZE(), then use ARRAY_SIZE(ad7606_channels_16bit) here instead of 9. > + .oversampling_avail = ad7606_oversampling_avail, > + .oversampling_num = ARRAY_SIZE(ad7606_oversampling_avail), > + }, > + [ID_AD7606C_18] = { > + .channels = ad7606_channels_18bit, > + .num_channels = 9, > + .oversampling_avail = ad7606_oversampling_avail, > + .oversampling_num = ARRAY_SIZE(ad7606_oversampling_avail), > + }, > [ID_AD7616] = { > .channels = ad7616_channels, > .num_channels = 17, > @@ -581,7 +662,122 @@ static const struct iio_trigger_ops ad7606_trigger_ops = { > .validate_device = iio_trigger_validate_own_device, > }; > > -static int ad7606_sw_mode_setup(struct iio_dev *indio_dev) > +static void ad7606c_18_chan_setup(struct ad7606_state *st, int ch, > + bool bipolar, bool differential) > +{ > + struct ad7606_chan_scale *cs = &st->chan_scales[ch]; > + > + if (differential) { > + cs->scale_avail = > + ad7606c_18_scale_differential_bipolar_avail; > + cs->num_scales = > + ARRAY_SIZE(ad7606c_18_scale_differential_bipolar_avail); > + /* Bipolar differential ranges start at 8 (b1000) */ > + cs->reg_offset = 8; > + cs->range = 1; > + } else if (bipolar) { > + cs->scale_avail = > + ad7606c_18_scale_single_ended_bipolar_avail; > + cs->num_scales = > + ARRAY_SIZE(ad7606c_18_scale_single_ended_bipolar_avail); I guess cs->reg_offset is 0 for this one? > + cs->range = 3; > + } else { > + cs->scale_avail = > + ad7606c_18_scale_single_ended_unipolar_avail; > + cs->num_scales = > + ARRAY_SIZE(ad7606c_18_scale_single_ended_unipolar_avail); > + /* Unipolar single-ended ranges start at 5 (b0101) */ > + cs->reg_offset = 5; > + cs->range = 1; > + } > +} > + > +static void ad7606c_16_chan_setup(struct ad7606_state *st, int ch, > + bool bipolar, bool differential) > +{ > + struct ad7606_chan_scale *cs = &st->chan_scales[ch]; > + > + if (differential) { > + cs->scale_avail = > + ad7606c_16_scale_differential_bipolar_avail; > + cs->num_scales = > + ARRAY_SIZE(ad7606c_16_scale_differential_bipolar_avail); > + /* Bipolar differential ranges start at 8 (b1000) */ > + cs->reg_offset = 8; > + cs->range = 1; > + } else if (bipolar) { > + cs->scale_avail = > + ad7606c_16_scale_single_ended_bipolar_avail; > + cs->num_scales = > + ARRAY_SIZE(ad7606c_16_scale_single_ended_bipolar_avail); > + cs->range = 3; > + } else { > + cs->scale_avail = > + ad7606c_16_scale_single_ended_unipolar_avail; > + cs->num_scales = > + ARRAY_SIZE(ad7606c_16_scale_single_ended_unipolar_avail); > + /* Unipolar single-ended ranges start at 5 (b0101) */ > + cs->reg_offset = 5; > + cs->range = 1; > + } > +} > + > +static int ad7606c_sw_mode_setup_channels(struct iio_dev *indio_dev, > + ad7606c_chan_setup_cb_t chan_setup_cb) > +{ > + unsigned int num_channels = indio_dev->num_channels - 1; > + struct ad7606_state *st = iio_priv(indio_dev); > + bool chan_configured[AD760X_MAX_CHANNELS] = {}; > + struct device *dev = st->dev; > + int ret; > + u32 ch; > + > + /* We need to hook this first */ Comment would be more useful if it said why. > + ret = st->bops->sw_mode_config(indio_dev); > + if (ret) > + return ret; > + > + device_for_each_child_node_scoped(dev, child) { > + bool bipolar, differential; > + u32 pins[2]; > + > + ret = fwnode_property_read_u32(child, "reg", &ch); > + if (ret) > + continue; > + > + /* channel number (here) is from 1 to num_channels */ > + if (ch == 0 || ch > num_channels) { > + dev_warn(st->dev, > + "Invalid channel number (ignoring): %d\n", ch); > + continue; > + } > + > + bipolar = fwnode_property_present(child, "bipolar"); IIRC, fwnode_property_read_bool() is preferred for bool/flag properties. > + > + ret = fwnode_property_read_u32_array(child, "diff-channels", > + pins, ARRAY_SIZE(pins)); > + /* Channel is differential, if pins are the same as 'reg' */ > + if (ret == 0 && pins[0] == ch && pins[1] == ch) > + differential = true; > + else > + differential = false; Would probably better to error on bad pin numbers rather than default to not differential. > + > + ch--; > + > + chan_setup_cb(st, ch, bipolar, differential); > + chan_configured[ch] = true; > + } > + > + /* Apply default configuration to unconfigured (via DT) channels */ > + for (ch = 0; ch < num_channels; ch++) { > + if (!chan_configured[ch]) > + chan_setup_cb(st, ch, false, false); > + } > + > + return 0; > +} > + > +static int ad7606_sw_mode_setup(struct iio_dev *indio_dev, unsigned int id) > { > unsigned int num_channels = indio_dev->num_channels - 1; > struct ad7606_state *st = iio_priv(indio_dev); > @@ -596,17 +792,30 @@ static int ad7606_sw_mode_setup(struct iio_dev *indio_dev) > > indio_dev->info = &ad7606_info_sw_mode; > > - /* Scale of 0.076293 is only available in sw mode */ > - /* After reset, in software mode, ±10 V is set by default */ > - for (ch = 0; ch < num_channels; ch++) { > - struct ad7606_chan_scale *cs = &st->chan_scales[ch]; > + switch (id) { > + case ID_AD7606C_18: > + ret = ad7606c_sw_mode_setup_channels(indio_dev, > + ad7606c_18_chan_setup); > + break; > + case ID_AD7606C_16: > + ret = ad7606c_sw_mode_setup_channels(indio_dev, > + ad7606c_16_chan_setup); > + break; > + default: > + /* Scale of 0.076293 is only available in sw mode */ > + /* After reset, in software mode, ±10 V is set by default */ > + for (ch = 0; ch < num_channels; ch++) { > + struct ad7606_chan_scale *cs = &st->chan_scales[ch]; > + > + cs->scale_avail = ad7616_sw_scale_avail; > + cs->num_scales = ARRAY_SIZE(ad7616_sw_scale_avail); > + cs->range = 2; > + } > > - cs->scale_avail = ad7616_sw_scale_avail; > - cs->num_scales = ARRAY_SIZE(ad7616_sw_scale_avail); > - cs->range = 2; > + ret = st->bops->sw_mode_config(indio_dev); > + break; > } > > - ret = st->bops->sw_mode_config(indio_dev); > if (ret) > return ret; > > @@ -655,9 +864,16 @@ int ad7606_probe(struct device *dev, int irq, void __iomem *base_address, > st->oversampling = 1; > > cs = &st->chan_scales[0]; > - cs->range = 0; > - cs->scale_avail = ad7606_scale_avail; > - cs->num_scales = ARRAY_SIZE(ad7606_scale_avail); > + switch (id) { > + case ID_AD7606C_18: > + cs->scale_avail = ad7606_18bit_hw_scale_avail; > + cs->num_scales = ARRAY_SIZE(ad7606_18bit_hw_scale_avail); > + break; > + default: > + cs->scale_avail = ad7606_16bit_hw_scale_avail; > + cs->num_scales = ARRAY_SIZE(ad7606_16bit_hw_scale_avail); > + break; > + } > > ret = devm_regulator_get_enable(dev, "avcc"); > if (ret) > @@ -706,7 +922,7 @@ int ad7606_probe(struct device *dev, int irq, void __iomem *base_address, > st->write_scale = ad7606_write_scale_hw; > st->write_os = ad7606_write_os_hw; > > - ret = ad7606_sw_mode_setup(indio_dev); > + ret = ad7606_sw_mode_setup(indio_dev, id); > if (ret) > return ret; > > diff --git a/drivers/iio/adc/ad7606.h b/drivers/iio/adc/ad7606.h > index 2113ad460c0f..6b0897aa2dc7 100644 > --- a/drivers/iio/adc/ad7606.h > +++ b/drivers/iio/adc/ad7606.h > @@ -22,7 +22,7 @@ > .scan_type = { \ > .sign = 's', \ > .realbits = (bits), \ > - .storagebits = (bits), \ > + .storagebits = (bits) > 16 ? 32 : 16, \ > .endianness = IIO_CPU, \ > }, \ > } > @@ -45,7 +45,7 @@ > .scan_type = { \ > .sign = 's', \ > .realbits = (bits), \ > - .storagebits = (bits), \ > + .storagebits = (bits) > 16 ? 32 : 16, \ > .endianness = IIO_CPU, \ > }, \ > } > @@ -88,6 +88,8 @@ struct ad7606_chip_info { > * such that it can be read via the 'read_avail' hook > * @num_scales number of elements stored in the scale_avail array > * @range voltage range selection, selects which scale to apply > + * @reg_offset offset for the register value, to be applied when > + * writing the value of 'range' to the register value > */ > struct ad7606_chan_scale { > #define AD760X_MAX_SCALE_SHOW (AD760X_MAX_CHANNELS * 2) > @@ -95,6 +97,7 @@ struct ad7606_chan_scale { > int scale_avail_show[AD760X_MAX_SCALE_SHOW]; > unsigned int num_scales; > unsigned int range; > + unsigned int reg_offset; > }; > > /** > @@ -151,9 +154,13 @@ struct ad7606_state { > /* > * DMA (thus cache coherency maintenance) may require the > * transfer buffers to live in their own cache lines. > - * 16 * 16-bit samples + 64-bit timestamp > + * 16 * 16-bit samples + 64-bit timestamp - for AD7616 > + * 8 * 32-bit samples + 64-bit timestamp - for AD7616C-18 (and similar) > */ > - unsigned short data[20] __aligned(IIO_DMA_MINALIGN); > + union { > + unsigned short d16[20]; > + unsigned int d32[10]; > + } data __aligned(IIO_DMA_MINALIGN); > __be16 d16[2]; > }; > > @@ -192,6 +199,8 @@ enum ad7606_supported_device_ids { > ID_AD7606_6, > ID_AD7606_4, > ID_AD7606B, > + ID_AD7606C_16, > + ID_AD7606C_18, > ID_AD7616, > }; > > diff --git a/drivers/iio/adc/ad7606_spi.c b/drivers/iio/adc/ad7606_spi.c > index e00f58a6a0e9..b8d630ad156d 100644 > --- a/drivers/iio/adc/ad7606_spi.c > +++ b/drivers/iio/adc/ad7606_spi.c > @@ -77,6 +77,18 @@ static const struct iio_chan_spec ad7606b_sw_channels[] = { > AD7606_SW_CHANNEL(7, 16), > }; > > +static const struct iio_chan_spec ad7606c_18_sw_channels[] = { > + IIO_CHAN_SOFT_TIMESTAMP(8), > + AD7606_SW_CHANNEL(0, 18), > + AD7606_SW_CHANNEL(1, 18), > + AD7606_SW_CHANNEL(2, 18), > + AD7606_SW_CHANNEL(3, 18), > + AD7606_SW_CHANNEL(4, 18), > + AD7606_SW_CHANNEL(5, 18), > + AD7606_SW_CHANNEL(6, 18), > + AD7606_SW_CHANNEL(7, 18), > +}; > + > static const unsigned int ad7606B_oversampling_avail[9] = { > 1, 2, 4, 8, 16, 32, 64, 128, 256 > }; > @@ -120,6 +132,19 @@ static int ad7606_spi_read_block(struct device *dev, > return 0; > } > > +static int ad7606_spi_read_block18to32(struct device *dev, > + int count, void *buf) > +{ > + struct spi_device *spi = to_spi_device(dev); > + struct spi_transfer xfer = { > + .bits_per_word = 18, > + .len = count, Isn't count the number of words? .len needs to be the number of bytes, so 4 * count. > + .rx_buf = buf, > + }; > + > + return spi_sync_transfer(spi, &xfer, 1); > +} > + > static int ad7606_spi_reg_read(struct ad7606_state *st, unsigned int addr) > { > struct spi_device *spi = to_spi_device(st->dev); > @@ -283,6 +308,19 @@ static int ad7606B_sw_mode_config(struct iio_dev *indio_dev) > return 0; > } > > +static int ad7606c_18_sw_mode_config(struct iio_dev *indio_dev) > +{ > + int ret; > + > + ret = ad7606B_sw_mode_config(indio_dev); > + if (ret) > + return ret; > + > + indio_dev->channels = ad7606c_18_sw_channels; > + > + return 0; > +} > + > static const struct ad7606_bus_ops ad7606_spi_bops = { > .read_block = ad7606_spi_read_block, > }; > @@ -305,6 +343,15 @@ static const struct ad7606_bus_ops ad7606B_spi_bops = { > .sw_mode_config = ad7606B_sw_mode_config, > }; > > +static const struct ad7606_bus_ops ad7606c_18_spi_bops = { > + .read_block = ad7606_spi_read_block18to32, > + .reg_read = ad7606_spi_reg_read, > + .reg_write = ad7606_spi_reg_write, > + .write_mask = ad7606_spi_write_mask, > + .rd_wr_cmd = ad7606B_spi_rd_wr_cmd, > + .sw_mode_config = ad7606c_18_sw_mode_config, > +}; > + > static int ad7606_spi_probe(struct spi_device *spi) > { > const struct spi_device_id *id = spi_get_device_id(spi); > @@ -315,8 +362,12 @@ static int ad7606_spi_probe(struct spi_device *spi) > bops = &ad7616_spi_bops; > break; > case ID_AD7606B: > + case ID_AD7606C_16: > bops = &ad7606B_spi_bops; > break; > + case ID_AD7606C_18: > + bops = &ad7606c_18_spi_bops; > + break; > default: > bops = &ad7606_spi_bops; > break; > @@ -333,6 +384,8 @@ static const struct spi_device_id ad7606_id_table[] = { > { "ad7606-6", ID_AD7606_6 }, > { "ad7606-8", ID_AD7606_8 }, > { "ad7606b", ID_AD7606B }, > + { "ad7606c-16", ID_AD7606C_16 }, > + { "ad7606c-18", ID_AD7606C_18 }, > { "ad7616", ID_AD7616 }, > { } > }; > @@ -344,6 +397,8 @@ static const struct of_device_id ad7606_of_match[] = { > { .compatible = "adi,ad7606-6" }, > { .compatible = "adi,ad7606-8" }, > { .compatible = "adi,ad7606b" }, > + { .compatible = "adi,ad7606c-16" }, > + { .compatible = "adi,ad7606c-18" }, > { .compatible = "adi,ad7616" }, > { } > };
On Fri, Sep 6, 2024 at 2:30 AM David Lechner <dlechner@baylibre.com> wrote: > > On 9/5/24 3:24 AM, Alexandru Ardelean wrote: > > The AD7606C-16 and AD7606C-18 are pretty similar with the AD7606B. > > The main difference between AD7606C-16 & AD7606C-18 is the precision in > > bits (16 vs 18). > > Because of that, some scales need to be defined for the 18-bit variants, as > > they need to be computed against 2**18 (vs 2**16 for the 16 bit-variants). > > > > Because the AD7606C-16,18 also supports bipolar & differential channels, > > for SW-mode, the default range of 10 V or ±10V should be set at probe. > > On reset, the default range (in the registers) is set to value 0x3 which > > corresponds to '±10 V single-ended range', regardless of bipolar or > > differential configuration. > > > > Aside from the scale/ranges, the AD7606C-16 is similar to the AD7606B. > > > > The AD7606C-18 variant offers 18-bit precision. Because of this, the > > requirement to use this chip is that the SPI controller supports padding > > of 18-bit sequences to 32-bit arrays. > > > > Datasheet links: > > https://www.analog.com/media/en/technical-documentation/data-sheets/ad7606c-16.pdf > > https://www.analog.com/media/en/technical-documentation/data-sheets/ad7606c-18.pdf > > > > Signed-off-by: Alexandru Ardelean <aardelean@baylibre.com> > > --- > > drivers/iio/adc/ad7606.c | 266 +++++++++++++++++++++++++++++++---- > > drivers/iio/adc/ad7606.h | 17 ++- > > drivers/iio/adc/ad7606_spi.c | 55 ++++++++ > > 3 files changed, 309 insertions(+), 29 deletions(-) > > > > diff --git a/drivers/iio/adc/ad7606.c b/drivers/iio/adc/ad7606.c > > index 4c3fbb28f790..999c4411859e 100644 > > --- a/drivers/iio/adc/ad7606.c > > +++ b/drivers/iio/adc/ad7606.c > > @@ -28,14 +28,44 @@ > > > > #include "ad7606.h" > > > > +typedef void (*ad7606c_chan_setup_cb_t)(struct ad7606_state *st, int ch, > > + bool bipolar, bool differential); > > + > > /* > > * Scales are computed as 5000/32768 and 10000/32768 respectively, > > * so that when applied to the raw values they provide mV values > > */ > > -static const unsigned int ad7606_scale_avail[2] = { > > +static const unsigned int ad7606_16bit_hw_scale_avail[2] = { > > 152588, 305176 > > }; > > > > +static const unsigned int ad7606_18bit_hw_scale_avail[2] = { > > + 38147, 76294 > > +}; > > + > > +static const unsigned int ad7606c_16_scale_single_ended_unipolar_avail[3] = { > > + 76294, 152588, 190735, > > +}; > > + > > +static const unsigned int ad7606c_16_scale_single_ended_bipolar_avail[5] = { > > + 76294, 152588, 190735, 305176, 381470 > > +}; > > + > > +static const unsigned int ad7606c_16_scale_differential_bipolar_avail[4] = { > > + 152588, 305176, 381470, 610352 > > +}; > > + > > +static const unsigned int ad7606c_18_scale_single_ended_unipolar_avail[3] = { > > + 19073, 38147, 47684 > > +}; > > + > > +static const unsigned int ad7606c_18_scale_single_ended_bipolar_avail[5] = { > > + 19073, 38147, 47684, 76294, 95367 > > +}; > > + > > +static const unsigned int ad7606c_18_scale_differential_bipolar_avail[4] = { > > + 38147, 76294, 95367, 152588 > > +}; > > > > static const unsigned int ad7616_sw_scale_avail[3] = { > > 76293, 152588, 305176 > > @@ -82,11 +112,19 @@ static int ad7606_reg_access(struct iio_dev *indio_dev, > > } > > } > > > > -static int ad7606_read_samples(struct ad7606_state *st) > > +static int ad7606_read_samples(struct ad7606_state *st, bool sign_extend_samples) > > { > > + unsigned int storagebits = st->chip_info->channels[1].scan_type.storagebits; > > Why [1]? Sure, they are all the same, but [0] would seem less arbitrary. [0] is the timestamp channel. > > > unsigned int num = st->chip_info->num_channels - 1; > > - u16 *data = st->data; > > - int ret; > > + u32 *data32 = st->data.d32; > > + u16 *data16 = st->data.d16; > > + void *data; > > + int i, ret; > > + > > + if (storagebits > 16) > > + data = data32; > > + else > > + data = data16; > > > > /* > > * The frstdata signal is set to high while and after reading the sample > > @@ -108,11 +146,25 @@ static int ad7606_read_samples(struct ad7606_state *st) > > return -EIO; > > } > > > > - data++; > > + if (storagebits > 16) > > + data32++; > > + else > > + data16++; > > num--; > > } > > > > - return st->bops->read_block(st->dev, num, data); > > + ret = st->bops->read_block(st->dev, num, data); > > Since data++ was removed, this looks broken now as well as the > other read_block() not visible in the diff. > > Maybe better to drop data32 and data16, keep the change of data > to void*, and change data++ to data += BITS_TO_BYTES(storagebits)? > > Although, all of this might be moot since it looks like this > needs to be rebased on [1]. > > [1]: https://git.kernel.org/pub/scm/linux/kernel/git/jic23/iio.git/commit/?h=fixes-togreg&id=90826e08468ba7fb35d8b39645b22d9e80004afe Oh. Omitted that patch. I forgot that fixes-togreg has a different cadence in another branch. > > > + if (ret) > > + return ret; > > + > > + if (storagebits == 16 || !sign_extend_samples) > > + return 0; > > + > > + /* For 18 bit samples, we need to sign-extend samples to 32 bits */ > > + for (i = 0; i < num; i++) > > + data32[i] = sign_extend32(data32[i], 17);> + > > + return 0; > > } > > > > static irqreturn_t ad7606_trigger_handler(int irq, void *p) > > @@ -124,11 +176,11 @@ static irqreturn_t ad7606_trigger_handler(int irq, void *p) > > > > guard(mutex)(&st->lock); > > > > - ret = ad7606_read_samples(st); > > + ret = ad7606_read_samples(st, true); > > Shouldn't the sign_extend parameter depend on if the data is unipolar or bipolar? [c1] Sign-extension is only needed for 18-bit samples. 16-bit samples are already properly sign(ed), but to 16-bits. It's a slight performance improvement, that may look quirky here. The idea here, is that for ad7606_scan_direct() we only need to sign-extend 1 sample of the 8 samples we get. And we need to sign-extend it to 32 bits regardless of it being 16-bit or 18-bit. In ad7606_trigger_handler(), the 16-bit samples were pushed as-is. Which means that we need to sign-extend the samples at least for 18-bits (as it is a new part) The question now becomes if we should sign-extend to 32-bits, 16-bit samples in ad7606_trigger_handler(), as that may break some ABI. > > > if (ret) > > goto error_ret; > > > > - iio_push_to_buffers_with_timestamp(indio_dev, st->data, > > + iio_push_to_buffers_with_timestamp(indio_dev, st->data.d16, > > iio_get_time_ns(indio_dev)); > > error_ret: > > iio_trigger_notify_done(indio_dev->trig); > > @@ -142,6 +194,7 @@ static int ad7606_scan_direct(struct iio_dev *indio_dev, unsigned int ch, > > int *val) > > { > > struct ad7606_state *st = iio_priv(indio_dev); > > + unsigned int storagebits = st->chip_info->channels[1].scan_type.storagebits; > > int ret; > > > > gpiod_set_value(st->gpio_convst, 1); > > @@ -152,9 +205,13 @@ static int ad7606_scan_direct(struct iio_dev *indio_dev, unsigned int ch, > > goto error_ret; > > } > > > > - ret = ad7606_read_samples(st); > > - if (ret == 0) > > - *val = sign_extend32(st->data[ch], 15); > > + ret = ad7606_read_samples(st, false); > > Why not let ad7606_read_samples() do the sign extending since > it can do that now? Related to comment [c1] > > > + if (ret == 0) { > > + if (storagebits > 16) > > + *val = sign_extend32(st->data.d32[ch], 17); > > + else > > + *val = sign_extend32(st->data.d16[ch], 15); > > + } > > > > error_ret: > > gpiod_set_value(st->gpio_convst, 0); > > @@ -267,7 +324,7 @@ static int ad7606_write_raw(struct iio_dev *indio_dev, > > ch = chan->address; > > cs = &st->chan_scales[ch]; > > i = find_closest(val2, cs->scale_avail, cs->num_scales); > > - ret = st->write_scale(indio_dev, ch, i); > > + ret = st->write_scale(indio_dev, ch, i + cs->reg_offset); > > if (ret < 0) > > return ret; > > cs->range = i; > > @@ -350,6 +407,18 @@ static const struct iio_chan_spec ad7606_channels_16bit[] = { > > AD7606_CHANNEL(7, 16), > > }; > > > > +static const struct iio_chan_spec ad7606_channels_18bit[] = { > > + IIO_CHAN_SOFT_TIMESTAMP(8), > > + AD7606_CHANNEL(0, 18), > > + AD7606_CHANNEL(1, 18), > > + AD7606_CHANNEL(2, 18), > > + AD7606_CHANNEL(3, 18), > > + AD7606_CHANNEL(4, 18), > > + AD7606_CHANNEL(5, 18), > > + AD7606_CHANNEL(6, 18), > > + AD7606_CHANNEL(7, 18), > > +}; > > + > > /* > > * The current assumption that this driver makes for AD7616, is that it's > > * working in Hardware Mode with Serial, Burst and Sequencer modes activated. > > @@ -410,6 +479,18 @@ static const struct ad7606_chip_info ad7606_chip_info_tbl[] = { > > .oversampling_avail = ad7606_oversampling_avail, > > .oversampling_num = ARRAY_SIZE(ad7606_oversampling_avail), > > }, > > + [ID_AD7606C_16] = { > > + .channels = ad7606_channels_16bit, > > + .num_channels = 9, > > Could be nice to have a cleanup patch before this to convert others to > use ARRAY_SIZE(), then use ARRAY_SIZE(ad7606_channels_16bit) here > instead of 9. Ack. > > > + .oversampling_avail = ad7606_oversampling_avail, > > + .oversampling_num = ARRAY_SIZE(ad7606_oversampling_avail), > > + }, > > + [ID_AD7606C_18] = { > > + .channels = ad7606_channels_18bit, > > + .num_channels = 9, > > + .oversampling_avail = ad7606_oversampling_avail, > > + .oversampling_num = ARRAY_SIZE(ad7606_oversampling_avail), > > + }, > > [ID_AD7616] = { > > .channels = ad7616_channels, > > .num_channels = 17, > > @@ -581,7 +662,122 @@ static const struct iio_trigger_ops ad7606_trigger_ops = { > > .validate_device = iio_trigger_validate_own_device, > > }; > > > > -static int ad7606_sw_mode_setup(struct iio_dev *indio_dev) > > +static void ad7606c_18_chan_setup(struct ad7606_state *st, int ch, > > + bool bipolar, bool differential) > > +{ > > + struct ad7606_chan_scale *cs = &st->chan_scales[ch]; > > + > > + if (differential) { > > + cs->scale_avail = > > + ad7606c_18_scale_differential_bipolar_avail; > > + cs->num_scales = > > + ARRAY_SIZE(ad7606c_18_scale_differential_bipolar_avail); > > + /* Bipolar differential ranges start at 8 (b1000) */ > > + cs->reg_offset = 8; > > + cs->range = 1; > > + } else if (bipolar) { > > + cs->scale_avail = > > + ad7606c_18_scale_single_ended_bipolar_avail; > > + cs->num_scales = > > + ARRAY_SIZE(ad7606c_18_scale_single_ended_bipolar_avail); > > I guess cs->reg_offset is 0 for this one? Yes. I will make it explicit. > > > + cs->range = 3; > > + } else { > > + cs->scale_avail = > > + ad7606c_18_scale_single_ended_unipolar_avail; > > + cs->num_scales = > > + ARRAY_SIZE(ad7606c_18_scale_single_ended_unipolar_avail); > > + /* Unipolar single-ended ranges start at 5 (b0101) */ > > + cs->reg_offset = 5; > > + cs->range = 1; > > + } > > +} > > + > > +static void ad7606c_16_chan_setup(struct ad7606_state *st, int ch, > > + bool bipolar, bool differential) > > +{ > > + struct ad7606_chan_scale *cs = &st->chan_scales[ch]; > > + > > + if (differential) { > > + cs->scale_avail = > > + ad7606c_16_scale_differential_bipolar_avail; > > + cs->num_scales = > > + ARRAY_SIZE(ad7606c_16_scale_differential_bipolar_avail); > > + /* Bipolar differential ranges start at 8 (b1000) */ > > + cs->reg_offset = 8; > > + cs->range = 1; > > + } else if (bipolar) { > > + cs->scale_avail = > > + ad7606c_16_scale_single_ended_bipolar_avail; > > + cs->num_scales = > > + ARRAY_SIZE(ad7606c_16_scale_single_ended_bipolar_avail); > > + cs->range = 3; > > + } else { > > + cs->scale_avail = > > + ad7606c_16_scale_single_ended_unipolar_avail; > > + cs->num_scales = > > + ARRAY_SIZE(ad7606c_16_scale_single_ended_unipolar_avail); > > + /* Unipolar single-ended ranges start at 5 (b0101) */ > > + cs->reg_offset = 5; > > + cs->range = 1; > > + } > > +} > > + > > +static int ad7606c_sw_mode_setup_channels(struct iio_dev *indio_dev, > > + ad7606c_chan_setup_cb_t chan_setup_cb) > > +{ > > + unsigned int num_channels = indio_dev->num_channels - 1; > > + struct ad7606_state *st = iio_priv(indio_dev); > > + bool chan_configured[AD760X_MAX_CHANNELS] = {}; > > + struct device *dev = st->dev; > > + int ret; > > + u32 ch; > > + > > + /* We need to hook this first */ > > Comment would be more useful if it said why. Ack. Will add. > > > + ret = st->bops->sw_mode_config(indio_dev); > > + if (ret) > > + return ret; > > + > > + device_for_each_child_node_scoped(dev, child) { > > + bool bipolar, differential; > > + u32 pins[2]; > > + > > + ret = fwnode_property_read_u32(child, "reg", &ch); > > + if (ret) > > + continue; > > + > > + /* channel number (here) is from 1 to num_channels */ > > + if (ch == 0 || ch > num_channels) { > > + dev_warn(st->dev, > > + "Invalid channel number (ignoring): %d\n", ch); > > + continue; > > + } > > + > > + bipolar = fwnode_property_present(child, "bipolar"); > > IIRC, fwnode_property_read_bool() is preferred for bool/flag properties. Ack. > > > + > > + ret = fwnode_property_read_u32_array(child, "diff-channels", > > + pins, ARRAY_SIZE(pins)); > > + /* Channel is differential, if pins are the same as 'reg' */ > > + if (ret == 0 && pins[0] == ch && pins[1] == ch) > > + differential = true; > > + else > > + differential = false; > > Would probably better to error on bad pin numbers rather than default to > not differential. No strong preference from my side. Will implement failure/error case. > > > + > > + ch--; > > + > > + chan_setup_cb(st, ch, bipolar, differential); > > + chan_configured[ch] = true; > > + } > > + > > + /* Apply default configuration to unconfigured (via DT) channels */ > > + for (ch = 0; ch < num_channels; ch++) { > > + if (!chan_configured[ch]) > > + chan_setup_cb(st, ch, false, false); > > + } > > + > > + return 0; > > +} > > + > > +static int ad7606_sw_mode_setup(struct iio_dev *indio_dev, unsigned int id) > > { > > unsigned int num_channels = indio_dev->num_channels - 1; > > struct ad7606_state *st = iio_priv(indio_dev); > > @@ -596,17 +792,30 @@ static int ad7606_sw_mode_setup(struct iio_dev *indio_dev) > > > > indio_dev->info = &ad7606_info_sw_mode; > > > > - /* Scale of 0.076293 is only available in sw mode */ > > - /* After reset, in software mode, ±10 V is set by default */ > > - for (ch = 0; ch < num_channels; ch++) { > > - struct ad7606_chan_scale *cs = &st->chan_scales[ch]; > > + switch (id) { > > + case ID_AD7606C_18: > > + ret = ad7606c_sw_mode_setup_channels(indio_dev, > > + ad7606c_18_chan_setup); > > + break; > > + case ID_AD7606C_16: > > + ret = ad7606c_sw_mode_setup_channels(indio_dev, > > + ad7606c_16_chan_setup); > > + break; > > + default: > > + /* Scale of 0.076293 is only available in sw mode */ > > + /* After reset, in software mode, ±10 V is set by default */ > > + for (ch = 0; ch < num_channels; ch++) { > > + struct ad7606_chan_scale *cs = &st->chan_scales[ch]; > > + > > + cs->scale_avail = ad7616_sw_scale_avail; > > + cs->num_scales = ARRAY_SIZE(ad7616_sw_scale_avail); > > + cs->range = 2; > > + } > > > > - cs->scale_avail = ad7616_sw_scale_avail; > > - cs->num_scales = ARRAY_SIZE(ad7616_sw_scale_avail); > > - cs->range = 2; > > + ret = st->bops->sw_mode_config(indio_dev); > > + break; > > } > > > > - ret = st->bops->sw_mode_config(indio_dev); > > if (ret) > > return ret; > > > > @@ -655,9 +864,16 @@ int ad7606_probe(struct device *dev, int irq, void __iomem *base_address, > > st->oversampling = 1; > > > > cs = &st->chan_scales[0]; > > - cs->range = 0; > > - cs->scale_avail = ad7606_scale_avail; > > - cs->num_scales = ARRAY_SIZE(ad7606_scale_avail); > > + switch (id) { > > + case ID_AD7606C_18: > > + cs->scale_avail = ad7606_18bit_hw_scale_avail; > > + cs->num_scales = ARRAY_SIZE(ad7606_18bit_hw_scale_avail); > > + break; > > + default: > > + cs->scale_avail = ad7606_16bit_hw_scale_avail; > > + cs->num_scales = ARRAY_SIZE(ad7606_16bit_hw_scale_avail); > > + break; > > + } > > > > ret = devm_regulator_get_enable(dev, "avcc"); > > if (ret) > > @@ -706,7 +922,7 @@ int ad7606_probe(struct device *dev, int irq, void __iomem *base_address, > > st->write_scale = ad7606_write_scale_hw; > > st->write_os = ad7606_write_os_hw; > > > > - ret = ad7606_sw_mode_setup(indio_dev); > > + ret = ad7606_sw_mode_setup(indio_dev, id); > > if (ret) > > return ret; > > > > diff --git a/drivers/iio/adc/ad7606.h b/drivers/iio/adc/ad7606.h > > index 2113ad460c0f..6b0897aa2dc7 100644 > > --- a/drivers/iio/adc/ad7606.h > > +++ b/drivers/iio/adc/ad7606.h > > @@ -22,7 +22,7 @@ > > .scan_type = { \ > > .sign = 's', \ > > .realbits = (bits), \ > > - .storagebits = (bits), \ > > + .storagebits = (bits) > 16 ? 32 : 16, \ > > .endianness = IIO_CPU, \ > > }, \ > > } > > @@ -45,7 +45,7 @@ > > .scan_type = { \ > > .sign = 's', \ > > .realbits = (bits), \ > > - .storagebits = (bits), \ > > + .storagebits = (bits) > 16 ? 32 : 16, \ > > .endianness = IIO_CPU, \ > > }, \ > > } > > @@ -88,6 +88,8 @@ struct ad7606_chip_info { > > * such that it can be read via the 'read_avail' hook > > * @num_scales number of elements stored in the scale_avail array > > * @range voltage range selection, selects which scale to apply > > + * @reg_offset offset for the register value, to be applied when > > + * writing the value of 'range' to the register value > > */ > > struct ad7606_chan_scale { > > #define AD760X_MAX_SCALE_SHOW (AD760X_MAX_CHANNELS * 2) > > @@ -95,6 +97,7 @@ struct ad7606_chan_scale { > > int scale_avail_show[AD760X_MAX_SCALE_SHOW]; > > unsigned int num_scales; > > unsigned int range; > > + unsigned int reg_offset; > > }; > > > > /** > > @@ -151,9 +154,13 @@ struct ad7606_state { > > /* > > * DMA (thus cache coherency maintenance) may require the > > * transfer buffers to live in their own cache lines. > > - * 16 * 16-bit samples + 64-bit timestamp > > + * 16 * 16-bit samples + 64-bit timestamp - for AD7616 > > + * 8 * 32-bit samples + 64-bit timestamp - for AD7616C-18 (and similar) > > */ > > - unsigned short data[20] __aligned(IIO_DMA_MINALIGN); > > + union { > > + unsigned short d16[20]; > > + unsigned int d32[10]; > > + } data __aligned(IIO_DMA_MINALIGN); > > __be16 d16[2]; > > }; > > > > @@ -192,6 +199,8 @@ enum ad7606_supported_device_ids { > > ID_AD7606_6, > > ID_AD7606_4, > > ID_AD7606B, > > + ID_AD7606C_16, > > + ID_AD7606C_18, > > ID_AD7616, > > }; > > > > diff --git a/drivers/iio/adc/ad7606_spi.c b/drivers/iio/adc/ad7606_spi.c > > index e00f58a6a0e9..b8d630ad156d 100644 > > --- a/drivers/iio/adc/ad7606_spi.c > > +++ b/drivers/iio/adc/ad7606_spi.c > > @@ -77,6 +77,18 @@ static const struct iio_chan_spec ad7606b_sw_channels[] = { > > AD7606_SW_CHANNEL(7, 16), > > }; > > > > +static const struct iio_chan_spec ad7606c_18_sw_channels[] = { > > + IIO_CHAN_SOFT_TIMESTAMP(8), > > + AD7606_SW_CHANNEL(0, 18), > > + AD7606_SW_CHANNEL(1, 18), > > + AD7606_SW_CHANNEL(2, 18), > > + AD7606_SW_CHANNEL(3, 18), > > + AD7606_SW_CHANNEL(4, 18), > > + AD7606_SW_CHANNEL(5, 18), > > + AD7606_SW_CHANNEL(6, 18), > > + AD7606_SW_CHANNEL(7, 18), > > +}; > > + > > static const unsigned int ad7606B_oversampling_avail[9] = { > > 1, 2, 4, 8, 16, 32, 64, 128, 256 > > }; > > @@ -120,6 +132,19 @@ static int ad7606_spi_read_block(struct device *dev, > > return 0; > > } > > > > +static int ad7606_spi_read_block18to32(struct device *dev, > > + int count, void *buf) > > +{ > > + struct spi_device *spi = to_spi_device(dev); > > + struct spi_transfer xfer = { > > + .bits_per_word = 18, > > + .len = count, > > Isn't count the number of words? .len needs to be the number > of bytes, so 4 * count. Oh. Hmm, I need to check. I assumed .len is the count of elements, not of bytes. > > > + .rx_buf = buf, > > + }; > > + > > + return spi_sync_transfer(spi, &xfer, 1); > > +} > > + > > static int ad7606_spi_reg_read(struct ad7606_state *st, unsigned int addr) > > { > > struct spi_device *spi = to_spi_device(st->dev); > > @@ -283,6 +308,19 @@ static int ad7606B_sw_mode_config(struct iio_dev *indio_dev) > > return 0; > > } > > > > +static int ad7606c_18_sw_mode_config(struct iio_dev *indio_dev) > > +{ > > + int ret; > > + > > + ret = ad7606B_sw_mode_config(indio_dev); > > + if (ret) > > + return ret; > > + > > + indio_dev->channels = ad7606c_18_sw_channels; > > + > > + return 0; > > +} > > + > > static const struct ad7606_bus_ops ad7606_spi_bops = { > > .read_block = ad7606_spi_read_block, > > }; > > @@ -305,6 +343,15 @@ static const struct ad7606_bus_ops ad7606B_spi_bops = { > > .sw_mode_config = ad7606B_sw_mode_config, > > }; > > > > +static const struct ad7606_bus_ops ad7606c_18_spi_bops = { > > + .read_block = ad7606_spi_read_block18to32, > > + .reg_read = ad7606_spi_reg_read, > > + .reg_write = ad7606_spi_reg_write, > > + .write_mask = ad7606_spi_write_mask, > > + .rd_wr_cmd = ad7606B_spi_rd_wr_cmd, > > + .sw_mode_config = ad7606c_18_sw_mode_config, > > +}; > > + > > static int ad7606_spi_probe(struct spi_device *spi) > > { > > const struct spi_device_id *id = spi_get_device_id(spi); > > @@ -315,8 +362,12 @@ static int ad7606_spi_probe(struct spi_device *spi) > > bops = &ad7616_spi_bops; > > break; > > case ID_AD7606B: > > + case ID_AD7606C_16: > > bops = &ad7606B_spi_bops; > > break; > > + case ID_AD7606C_18: > > + bops = &ad7606c_18_spi_bops; > > + break; > > default: > > bops = &ad7606_spi_bops; > > break; > > @@ -333,6 +384,8 @@ static const struct spi_device_id ad7606_id_table[] = { > > { "ad7606-6", ID_AD7606_6 }, > > { "ad7606-8", ID_AD7606_8 }, > > { "ad7606b", ID_AD7606B }, > > + { "ad7606c-16", ID_AD7606C_16 }, > > + { "ad7606c-18", ID_AD7606C_18 }, > > { "ad7616", ID_AD7616 }, > > { } > > }; > > @@ -344,6 +397,8 @@ static const struct of_device_id ad7606_of_match[] = { > > { .compatible = "adi,ad7606-6" }, > > { .compatible = "adi,ad7606-8" }, > > { .compatible = "adi,ad7606b" }, > > + { .compatible = "adi,ad7606c-16" }, > > + { .compatible = "adi,ad7606c-18" }, > > { .compatible = "adi,ad7616" }, > > { } > > }; >
On 9/6/24 12:34 AM, Alexandru Ardelean wrote: > On Fri, Sep 6, 2024 at 2:30 AM David Lechner <dlechner@baylibre.com> wrote: >> >> On 9/5/24 3:24 AM, Alexandru Ardelean wrote: >>> -static int ad7606_read_samples(struct ad7606_state *st) >>> +static int ad7606_read_samples(struct ad7606_state *st, bool sign_extend_samples) >>> { >>> + unsigned int storagebits = st->chip_info->channels[1].scan_type.storagebits; >> >> Why [1]? Sure, they are all the same, but [0] would seem less arbitrary. > > [0] is the timestamp channel. Oh, that's weird. First channel but last scan index!? >> >>> + if (ret) >>> + return ret; >>> + >>> + if (storagebits == 16 || !sign_extend_samples) >>> + return 0; >>> + >>> + /* For 18 bit samples, we need to sign-extend samples to 32 bits */ >>> + for (i = 0; i < num; i++) >>> + data32[i] = sign_extend32(data32[i], 17);> + >>> + return 0; >>> } >>> >>> static irqreturn_t ad7606_trigger_handler(int irq, void *p) >>> @@ -124,11 +176,11 @@ static irqreturn_t ad7606_trigger_handler(int irq, void *p) >>> >>> guard(mutex)(&st->lock); >>> >>> - ret = ad7606_read_samples(st); >>> + ret = ad7606_read_samples(st, true); >> >> Shouldn't the sign_extend parameter depend on if the data is unipolar or bipolar? > > [c1] > Sign-extension is only needed for 18-bit samples. > 16-bit samples are already properly sign(ed), but to 16-bits. > > It's a slight performance improvement, that may look quirky here. > The idea here, is that for ad7606_scan_direct() we only need to > sign-extend 1 sample of the 8 samples we get. > And we need to sign-extend it to 32 bits regardless of it being 16-bit > or 18-bit. > > In ad7606_trigger_handler(), the 16-bit samples were pushed as-is. > Which means that we need to sign-extend the samples at least for > 18-bits (as it is a new part) > The question now becomes if we should sign-extend to 32-bits, 16-bit > samples in ad7606_trigger_handler(), as that may break some ABI. > Sign extension should not be needed at all for buffered reads (that is what scan_type is for). So sign extension should only be needed for the direct read when returning a raw value via sysfs (raw read).
On Fri, Sep 6, 2024 at 4:33 PM David Lechner <dlechner@baylibre.com> wrote: > > On 9/6/24 12:34 AM, Alexandru Ardelean wrote: > > On Fri, Sep 6, 2024 at 2:30 AM David Lechner <dlechner@baylibre.com> wrote: > >> > >> On 9/5/24 3:24 AM, Alexandru Ardelean wrote: > > > >>> -static int ad7606_read_samples(struct ad7606_state *st) > >>> +static int ad7606_read_samples(struct ad7606_state *st, bool sign_extend_samples) > >>> { > >>> + unsigned int storagebits = st->chip_info->channels[1].scan_type.storagebits; > >> > >> Why [1]? Sure, they are all the same, but [0] would seem less arbitrary. > > > > [0] is the timestamp channel. > > Oh, that's weird. First channel but last scan index!? Yep ¯\_(ツ)_/¯ > > > >> > >>> + if (ret) > >>> + return ret; > >>> + > >>> + if (storagebits == 16 || !sign_extend_samples) > >>> + return 0; > >>> + > >>> + /* For 18 bit samples, we need to sign-extend samples to 32 bits */ > >>> + for (i = 0; i < num; i++) > >>> + data32[i] = sign_extend32(data32[i], 17);> + > >>> + return 0; > >>> } > >>> > >>> static irqreturn_t ad7606_trigger_handler(int irq, void *p) > >>> @@ -124,11 +176,11 @@ static irqreturn_t ad7606_trigger_handler(int irq, void *p) > >>> > >>> guard(mutex)(&st->lock); > >>> > >>> - ret = ad7606_read_samples(st); > >>> + ret = ad7606_read_samples(st, true); > >> > >> Shouldn't the sign_extend parameter depend on if the data is unipolar or bipolar? > > > > [c1] > > Sign-extension is only needed for 18-bit samples. > > 16-bit samples are already properly sign(ed), but to 16-bits. > > > > It's a slight performance improvement, that may look quirky here. > > The idea here, is that for ad7606_scan_direct() we only need to > > sign-extend 1 sample of the 8 samples we get. > > And we need to sign-extend it to 32 bits regardless of it being 16-bit > > or 18-bit. > > > > In ad7606_trigger_handler(), the 16-bit samples were pushed as-is. > > Which means that we need to sign-extend the samples at least for > > 18-bits (as it is a new part) > > The question now becomes if we should sign-extend to 32-bits, 16-bit > > samples in ad7606_trigger_handler(), as that may break some ABI. > > > > Sign extension should not be needed at all for buffered reads (that is > what scan_type is for). So sign extension should only be needed for > the direct read when returning a raw value via sysfs (raw read). ack; will remove it then from ad7606_read_samples()
On Fri, Sep 6, 2024 at 8:34 AM Alexandru Ardelean <aardelean@baylibre.com> wrote: > > On Fri, Sep 6, 2024 at 2:30 AM David Lechner <dlechner@baylibre.com> wrote: > > > > On 9/5/24 3:24 AM, Alexandru Ardelean wrote: > > > The AD7606C-16 and AD7606C-18 are pretty similar with the AD7606B. > > > The main difference between AD7606C-16 & AD7606C-18 is the precision in > > > bits (16 vs 18). > > > Because of that, some scales need to be defined for the 18-bit variants, as > > > they need to be computed against 2**18 (vs 2**16 for the 16 bit-variants). > > > > > > Because the AD7606C-16,18 also supports bipolar & differential channels, > > > for SW-mode, the default range of 10 V or ±10V should be set at probe. > > > On reset, the default range (in the registers) is set to value 0x3 which > > > corresponds to '±10 V single-ended range', regardless of bipolar or > > > differential configuration. > > > > > > Aside from the scale/ranges, the AD7606C-16 is similar to the AD7606B. > > > > > > The AD7606C-18 variant offers 18-bit precision. Because of this, the > > > requirement to use this chip is that the SPI controller supports padding > > > of 18-bit sequences to 32-bit arrays. > > > > > > Datasheet links: > > > https://www.analog.com/media/en/technical-documentation/data-sheets/ad7606c-16.pdf > > > https://www.analog.com/media/en/technical-documentation/data-sheets/ad7606c-18.pdf > > > > > > Signed-off-by: Alexandru Ardelean <aardelean@baylibre.com> > > > --- > > > drivers/iio/adc/ad7606.c | 266 +++++++++++++++++++++++++++++++---- > > > drivers/iio/adc/ad7606.h | 17 ++- > > > drivers/iio/adc/ad7606_spi.c | 55 ++++++++ > > > 3 files changed, 309 insertions(+), 29 deletions(-) > > > > > > diff --git a/drivers/iio/adc/ad7606.c b/drivers/iio/adc/ad7606.c > > > index 4c3fbb28f790..999c4411859e 100644 > > > --- a/drivers/iio/adc/ad7606.c > > > +++ b/drivers/iio/adc/ad7606.c > > > @@ -28,14 +28,44 @@ > > > > > > #include "ad7606.h" > > > > > > +typedef void (*ad7606c_chan_setup_cb_t)(struct ad7606_state *st, int ch, > > > + bool bipolar, bool differential); > > > + > > > /* > > > * Scales are computed as 5000/32768 and 10000/32768 respectively, > > > * so that when applied to the raw values they provide mV values > > > */ > > > -static const unsigned int ad7606_scale_avail[2] = { > > > +static const unsigned int ad7606_16bit_hw_scale_avail[2] = { > > > 152588, 305176 > > > }; > > > > > > +static const unsigned int ad7606_18bit_hw_scale_avail[2] = { > > > + 38147, 76294 > > > +}; > > > + > > > +static const unsigned int ad7606c_16_scale_single_ended_unipolar_avail[3] = { > > > + 76294, 152588, 190735, > > > +}; > > > + > > > +static const unsigned int ad7606c_16_scale_single_ended_bipolar_avail[5] = { > > > + 76294, 152588, 190735, 305176, 381470 > > > +}; > > > + > > > +static const unsigned int ad7606c_16_scale_differential_bipolar_avail[4] = { > > > + 152588, 305176, 381470, 610352 > > > +}; > > > + > > > +static const unsigned int ad7606c_18_scale_single_ended_unipolar_avail[3] = { > > > + 19073, 38147, 47684 > > > +}; > > > + > > > +static const unsigned int ad7606c_18_scale_single_ended_bipolar_avail[5] = { > > > + 19073, 38147, 47684, 76294, 95367 > > > +}; > > > + > > > +static const unsigned int ad7606c_18_scale_differential_bipolar_avail[4] = { > > > + 38147, 76294, 95367, 152588 > > > +}; > > > > > > static const unsigned int ad7616_sw_scale_avail[3] = { > > > 76293, 152588, 305176 > > > @@ -82,11 +112,19 @@ static int ad7606_reg_access(struct iio_dev *indio_dev, > > > } > > > } > > > > > > -static int ad7606_read_samples(struct ad7606_state *st) > > > +static int ad7606_read_samples(struct ad7606_state *st, bool sign_extend_samples) > > > { > > > + unsigned int storagebits = st->chip_info->channels[1].scan_type.storagebits; > > > > Why [1]? Sure, they are all the same, but [0] would seem less arbitrary. > > [0] is the timestamp channel. > > > > > > > unsigned int num = st->chip_info->num_channels - 1; > > > - u16 *data = st->data; > > > - int ret; > > > + u32 *data32 = st->data.d32; > > > + u16 *data16 = st->data.d16; > > > + void *data; > > > + int i, ret; > > > + > > > + if (storagebits > 16) > > > + data = data32; > > > + else > > > + data = data16; > > > > > > /* > > > * The frstdata signal is set to high while and after reading the sample > > > @@ -108,11 +146,25 @@ static int ad7606_read_samples(struct ad7606_state *st) > > > return -EIO; > > > } > > > > > > - data++; > > > + if (storagebits > 16) > > > + data32++; > > > + else > > > + data16++; > > > num--; > > > } > > > > > > - return st->bops->read_block(st->dev, num, data); > > > + ret = st->bops->read_block(st->dev, num, data); > > > > Since data++ was removed, this looks broken now as well as the > > other read_block() not visible in the diff. > > > > Maybe better to drop data32 and data16, keep the change of data > > to void*, and change data++ to data += BITS_TO_BYTES(storagebits)? > > > > Although, all of this might be moot since it looks like this > > needs to be rebased on [1]. > > > > [1]: https://git.kernel.org/pub/scm/linux/kernel/git/jic23/iio.git/commit/?h=fixes-togreg&id=90826e08468ba7fb35d8b39645b22d9e80004afe > > Oh. > Omitted that patch. > I forgot that fixes-togreg has a different cadence in another branch. > > > > > > + if (ret) > > > + return ret; > > > + > > > + if (storagebits == 16 || !sign_extend_samples) > > > + return 0; > > > + > > > + /* For 18 bit samples, we need to sign-extend samples to 32 bits */ > > > + for (i = 0; i < num; i++) > > > + data32[i] = sign_extend32(data32[i], 17);> + > > > + return 0; > > > } > > > > > > static irqreturn_t ad7606_trigger_handler(int irq, void *p) > > > @@ -124,11 +176,11 @@ static irqreturn_t ad7606_trigger_handler(int irq, void *p) > > > > > > guard(mutex)(&st->lock); > > > > > > - ret = ad7606_read_samples(st); > > > + ret = ad7606_read_samples(st, true); > > > > Shouldn't the sign_extend parameter depend on if the data is unipolar or bipolar? > > [c1] > Sign-extension is only needed for 18-bit samples. > 16-bit samples are already properly sign(ed), but to 16-bits. > > It's a slight performance improvement, that may look quirky here. > The idea here, is that for ad7606_scan_direct() we only need to > sign-extend 1 sample of the 8 samples we get. > And we need to sign-extend it to 32 bits regardless of it being 16-bit > or 18-bit. > > In ad7606_trigger_handler(), the 16-bit samples were pushed as-is. > Which means that we need to sign-extend the samples at least for > 18-bits (as it is a new part) > The question now becomes if we should sign-extend to 32-bits, 16-bit > samples in ad7606_trigger_handler(), as that may break some ABI. > > > > > > if (ret) > > > goto error_ret; > > > > > > - iio_push_to_buffers_with_timestamp(indio_dev, st->data, > > > + iio_push_to_buffers_with_timestamp(indio_dev, st->data.d16, > > > iio_get_time_ns(indio_dev)); > > > error_ret: > > > iio_trigger_notify_done(indio_dev->trig); > > > @@ -142,6 +194,7 @@ static int ad7606_scan_direct(struct iio_dev *indio_dev, unsigned int ch, > > > int *val) > > > { > > > struct ad7606_state *st = iio_priv(indio_dev); > > > + unsigned int storagebits = st->chip_info->channels[1].scan_type.storagebits; > > > int ret; > > > > > > gpiod_set_value(st->gpio_convst, 1); > > > @@ -152,9 +205,13 @@ static int ad7606_scan_direct(struct iio_dev *indio_dev, unsigned int ch, > > > goto error_ret; > > > } > > > > > > - ret = ad7606_read_samples(st); > > > - if (ret == 0) > > > - *val = sign_extend32(st->data[ch], 15); > > > + ret = ad7606_read_samples(st, false); > > > > Why not let ad7606_read_samples() do the sign extending since > > it can do that now? > > Related to comment [c1] > > > > > > + if (ret == 0) { > > > + if (storagebits > 16) > > > + *val = sign_extend32(st->data.d32[ch], 17); > > > + else > > > + *val = sign_extend32(st->data.d16[ch], 15); > > > + } > > > > > > error_ret: > > > gpiod_set_value(st->gpio_convst, 0); > > > @@ -267,7 +324,7 @@ static int ad7606_write_raw(struct iio_dev *indio_dev, > > > ch = chan->address; > > > cs = &st->chan_scales[ch]; > > > i = find_closest(val2, cs->scale_avail, cs->num_scales); > > > - ret = st->write_scale(indio_dev, ch, i); > > > + ret = st->write_scale(indio_dev, ch, i + cs->reg_offset); > > > if (ret < 0) > > > return ret; > > > cs->range = i; > > > @@ -350,6 +407,18 @@ static const struct iio_chan_spec ad7606_channels_16bit[] = { > > > AD7606_CHANNEL(7, 16), > > > }; > > > > > > +static const struct iio_chan_spec ad7606_channels_18bit[] = { > > > + IIO_CHAN_SOFT_TIMESTAMP(8), > > > + AD7606_CHANNEL(0, 18), > > > + AD7606_CHANNEL(1, 18), > > > + AD7606_CHANNEL(2, 18), > > > + AD7606_CHANNEL(3, 18), > > > + AD7606_CHANNEL(4, 18), > > > + AD7606_CHANNEL(5, 18), > > > + AD7606_CHANNEL(6, 18), > > > + AD7606_CHANNEL(7, 18), > > > +}; > > > + > > > /* > > > * The current assumption that this driver makes for AD7616, is that it's > > > * working in Hardware Mode with Serial, Burst and Sequencer modes activated. > > > @@ -410,6 +479,18 @@ static const struct ad7606_chip_info ad7606_chip_info_tbl[] = { > > > .oversampling_avail = ad7606_oversampling_avail, > > > .oversampling_num = ARRAY_SIZE(ad7606_oversampling_avail), > > > }, > > > + [ID_AD7606C_16] = { > > > + .channels = ad7606_channels_16bit, > > > + .num_channels = 9, > > > > Could be nice to have a cleanup patch before this to convert others to > > use ARRAY_SIZE(), then use ARRAY_SIZE(ad7606_channels_16bit) here > > instead of 9. > > Ack. I'll need to revert this to a NACK. "ad7606_channels{_16bit}" has 9 elements. But for some ADC variants, only 7 are used. So, num_channels is specified as 7 there, and references this same array. Which also explains why the timestamp channel is the first. > > > > > > + .oversampling_avail = ad7606_oversampling_avail, > > > + .oversampling_num = ARRAY_SIZE(ad7606_oversampling_avail), > > > + }, > > > + [ID_AD7606C_18] = { > > > + .channels = ad7606_channels_18bit, > > > + .num_channels = 9, Continuing here: I could make it ARRAY_SIZE(ad7606_channels_18bit), but that would make it less consistent with the other "num_channels" styles. And maybe there will be a 4 or 6 channel 18 bit part later. > > > + .oversampling_avail = ad7606_oversampling_avail, > > > + .oversampling_num = ARRAY_SIZE(ad7606_oversampling_avail), > > > + }, > > > [ID_AD7616] = { > > > .channels = ad7616_channels, > > > .num_channels = 17, > > > @@ -581,7 +662,122 @@ static const struct iio_trigger_ops ad7606_trigger_ops = { > > > .validate_device = iio_trigger_validate_own_device, > > > }; > > > > > > -static int ad7606_sw_mode_setup(struct iio_dev *indio_dev) > > > +static void ad7606c_18_chan_setup(struct ad7606_state *st, int ch, > > > + bool bipolar, bool differential) > > > +{ > > > + struct ad7606_chan_scale *cs = &st->chan_scales[ch]; > > > + > > > + if (differential) { > > > + cs->scale_avail = > > > + ad7606c_18_scale_differential_bipolar_avail; > > > + cs->num_scales = > > > + ARRAY_SIZE(ad7606c_18_scale_differential_bipolar_avail); > > > + /* Bipolar differential ranges start at 8 (b1000) */ > > > + cs->reg_offset = 8; > > > + cs->range = 1; > > > + } else if (bipolar) { > > > + cs->scale_avail = > > > + ad7606c_18_scale_single_ended_bipolar_avail; > > > + cs->num_scales = > > > + ARRAY_SIZE(ad7606c_18_scale_single_ended_bipolar_avail); > > > > I guess cs->reg_offset is 0 for this one? > > Yes. > I will make it explicit. > > > > > > + cs->range = 3; > > > + } else { > > > + cs->scale_avail = > > > + ad7606c_18_scale_single_ended_unipolar_avail; > > > + cs->num_scales = > > > + ARRAY_SIZE(ad7606c_18_scale_single_ended_unipolar_avail); > > > + /* Unipolar single-ended ranges start at 5 (b0101) */ > > > + cs->reg_offset = 5; > > > + cs->range = 1; > > > + } > > > +} > > > + > > > +static void ad7606c_16_chan_setup(struct ad7606_state *st, int ch, > > > + bool bipolar, bool differential) > > > +{ > > > + struct ad7606_chan_scale *cs = &st->chan_scales[ch]; > > > + > > > + if (differential) { > > > + cs->scale_avail = > > > + ad7606c_16_scale_differential_bipolar_avail; > > > + cs->num_scales = > > > + ARRAY_SIZE(ad7606c_16_scale_differential_bipolar_avail); > > > + /* Bipolar differential ranges start at 8 (b1000) */ > > > + cs->reg_offset = 8; > > > + cs->range = 1; > > > + } else if (bipolar) { > > > + cs->scale_avail = > > > + ad7606c_16_scale_single_ended_bipolar_avail; > > > + cs->num_scales = > > > + ARRAY_SIZE(ad7606c_16_scale_single_ended_bipolar_avail); > > > + cs->range = 3; > > > + } else { > > > + cs->scale_avail = > > > + ad7606c_16_scale_single_ended_unipolar_avail; > > > + cs->num_scales = > > > + ARRAY_SIZE(ad7606c_16_scale_single_ended_unipolar_avail); > > > + /* Unipolar single-ended ranges start at 5 (b0101) */ > > > + cs->reg_offset = 5; > > > + cs->range = 1; > > > + } > > > +} > > > + > > > +static int ad7606c_sw_mode_setup_channels(struct iio_dev *indio_dev, > > > + ad7606c_chan_setup_cb_t chan_setup_cb) > > > +{ > > > + unsigned int num_channels = indio_dev->num_channels - 1; > > > + struct ad7606_state *st = iio_priv(indio_dev); > > > + bool chan_configured[AD760X_MAX_CHANNELS] = {}; > > > + struct device *dev = st->dev; > > > + int ret; > > > + u32 ch; > > > + > > > + /* We need to hook this first */ > > > > Comment would be more useful if it said why. > > Ack. > Will add. > > > > > > + ret = st->bops->sw_mode_config(indio_dev); > > > + if (ret) > > > + return ret; > > > + > > > + device_for_each_child_node_scoped(dev, child) { > > > + bool bipolar, differential; > > > + u32 pins[2]; > > > + > > > + ret = fwnode_property_read_u32(child, "reg", &ch); > > > + if (ret) > > > + continue; > > > + > > > + /* channel number (here) is from 1 to num_channels */ > > > + if (ch == 0 || ch > num_channels) { > > > + dev_warn(st->dev, > > > + "Invalid channel number (ignoring): %d\n", ch); > > > + continue; > > > + } > > > + > > > + bipolar = fwnode_property_present(child, "bipolar"); > > > > IIRC, fwnode_property_read_bool() is preferred for bool/flag properties. > > Ack. > > > > > > + > > > + ret = fwnode_property_read_u32_array(child, "diff-channels", > > > + pins, ARRAY_SIZE(pins)); > > > + /* Channel is differential, if pins are the same as 'reg' */ > > > + if (ret == 0 && pins[0] == ch && pins[1] == ch) > > > + differential = true; > > > + else > > > + differential = false; > > > > Would probably better to error on bad pin numbers rather than default to > > not differential. > > No strong preference from my side. > Will implement failure/error case. > > > > > > + > > > + ch--; > > > + > > > + chan_setup_cb(st, ch, bipolar, differential); > > > + chan_configured[ch] = true; > > > + } > > > + > > > + /* Apply default configuration to unconfigured (via DT) channels */ > > > + for (ch = 0; ch < num_channels; ch++) { > > > + if (!chan_configured[ch]) > > > + chan_setup_cb(st, ch, false, false); > > > + } > > > + > > > + return 0; > > > +} > > > + > > > +static int ad7606_sw_mode_setup(struct iio_dev *indio_dev, unsigned int id) > > > { > > > unsigned int num_channels = indio_dev->num_channels - 1; > > > struct ad7606_state *st = iio_priv(indio_dev); > > > @@ -596,17 +792,30 @@ static int ad7606_sw_mode_setup(struct iio_dev *indio_dev) > > > > > > indio_dev->info = &ad7606_info_sw_mode; > > > > > > - /* Scale of 0.076293 is only available in sw mode */ > > > - /* After reset, in software mode, ±10 V is set by default */ > > > - for (ch = 0; ch < num_channels; ch++) { > > > - struct ad7606_chan_scale *cs = &st->chan_scales[ch]; > > > + switch (id) { > > > + case ID_AD7606C_18: > > > + ret = ad7606c_sw_mode_setup_channels(indio_dev, > > > + ad7606c_18_chan_setup); > > > + break; > > > + case ID_AD7606C_16: > > > + ret = ad7606c_sw_mode_setup_channels(indio_dev, > > > + ad7606c_16_chan_setup); > > > + break; > > > + default: > > > + /* Scale of 0.076293 is only available in sw mode */ > > > + /* After reset, in software mode, ±10 V is set by default */ > > > + for (ch = 0; ch < num_channels; ch++) { > > > + struct ad7606_chan_scale *cs = &st->chan_scales[ch]; > > > + > > > + cs->scale_avail = ad7616_sw_scale_avail; > > > + cs->num_scales = ARRAY_SIZE(ad7616_sw_scale_avail); > > > + cs->range = 2; > > > + } > > > > > > - cs->scale_avail = ad7616_sw_scale_avail; > > > - cs->num_scales = ARRAY_SIZE(ad7616_sw_scale_avail); > > > - cs->range = 2; > > > + ret = st->bops->sw_mode_config(indio_dev); > > > + break; > > > } > > > > > > - ret = st->bops->sw_mode_config(indio_dev); > > > if (ret) > > > return ret; > > > > > > @@ -655,9 +864,16 @@ int ad7606_probe(struct device *dev, int irq, void __iomem *base_address, > > > st->oversampling = 1; > > > > > > cs = &st->chan_scales[0]; > > > - cs->range = 0; > > > - cs->scale_avail = ad7606_scale_avail; > > > - cs->num_scales = ARRAY_SIZE(ad7606_scale_avail); > > > + switch (id) { > > > + case ID_AD7606C_18: > > > + cs->scale_avail = ad7606_18bit_hw_scale_avail; > > > + cs->num_scales = ARRAY_SIZE(ad7606_18bit_hw_scale_avail); > > > + break; > > > + default: > > > + cs->scale_avail = ad7606_16bit_hw_scale_avail; > > > + cs->num_scales = ARRAY_SIZE(ad7606_16bit_hw_scale_avail); > > > + break; > > > + } > > > > > > ret = devm_regulator_get_enable(dev, "avcc"); > > > if (ret) > > > @@ -706,7 +922,7 @@ int ad7606_probe(struct device *dev, int irq, void __iomem *base_address, > > > st->write_scale = ad7606_write_scale_hw; > > > st->write_os = ad7606_write_os_hw; > > > > > > - ret = ad7606_sw_mode_setup(indio_dev); > > > + ret = ad7606_sw_mode_setup(indio_dev, id); > > > if (ret) > > > return ret; > > > > > > diff --git a/drivers/iio/adc/ad7606.h b/drivers/iio/adc/ad7606.h > > > index 2113ad460c0f..6b0897aa2dc7 100644 > > > --- a/drivers/iio/adc/ad7606.h > > > +++ b/drivers/iio/adc/ad7606.h > > > @@ -22,7 +22,7 @@ > > > .scan_type = { \ > > > .sign = 's', \ > > > .realbits = (bits), \ > > > - .storagebits = (bits), \ > > > + .storagebits = (bits) > 16 ? 32 : 16, \ > > > .endianness = IIO_CPU, \ > > > }, \ > > > } > > > @@ -45,7 +45,7 @@ > > > .scan_type = { \ > > > .sign = 's', \ > > > .realbits = (bits), \ > > > - .storagebits = (bits), \ > > > + .storagebits = (bits) > 16 ? 32 : 16, \ > > > .endianness = IIO_CPU, \ > > > }, \ > > > } > > > @@ -88,6 +88,8 @@ struct ad7606_chip_info { > > > * such that it can be read via the 'read_avail' hook > > > * @num_scales number of elements stored in the scale_avail array > > > * @range voltage range selection, selects which scale to apply > > > + * @reg_offset offset for the register value, to be applied when > > > + * writing the value of 'range' to the register value > > > */ > > > struct ad7606_chan_scale { > > > #define AD760X_MAX_SCALE_SHOW (AD760X_MAX_CHANNELS * 2) > > > @@ -95,6 +97,7 @@ struct ad7606_chan_scale { > > > int scale_avail_show[AD760X_MAX_SCALE_SHOW]; > > > unsigned int num_scales; > > > unsigned int range; > > > + unsigned int reg_offset; > > > }; > > > > > > /** > > > @@ -151,9 +154,13 @@ struct ad7606_state { > > > /* > > > * DMA (thus cache coherency maintenance) may require the > > > * transfer buffers to live in their own cache lines. > > > - * 16 * 16-bit samples + 64-bit timestamp > > > + * 16 * 16-bit samples + 64-bit timestamp - for AD7616 > > > + * 8 * 32-bit samples + 64-bit timestamp - for AD7616C-18 (and similar) > > > */ > > > - unsigned short data[20] __aligned(IIO_DMA_MINALIGN); > > > + union { > > > + unsigned short d16[20]; > > > + unsigned int d32[10]; > > > + } data __aligned(IIO_DMA_MINALIGN); > > > __be16 d16[2]; > > > }; > > > > > > @@ -192,6 +199,8 @@ enum ad7606_supported_device_ids { > > > ID_AD7606_6, > > > ID_AD7606_4, > > > ID_AD7606B, > > > + ID_AD7606C_16, > > > + ID_AD7606C_18, > > > ID_AD7616, > > > }; > > > > > > diff --git a/drivers/iio/adc/ad7606_spi.c b/drivers/iio/adc/ad7606_spi.c > > > index e00f58a6a0e9..b8d630ad156d 100644 > > > --- a/drivers/iio/adc/ad7606_spi.c > > > +++ b/drivers/iio/adc/ad7606_spi.c > > > @@ -77,6 +77,18 @@ static const struct iio_chan_spec ad7606b_sw_channels[] = { > > > AD7606_SW_CHANNEL(7, 16), > > > }; > > > > > > +static const struct iio_chan_spec ad7606c_18_sw_channels[] = { > > > + IIO_CHAN_SOFT_TIMESTAMP(8), > > > + AD7606_SW_CHANNEL(0, 18), > > > + AD7606_SW_CHANNEL(1, 18), > > > + AD7606_SW_CHANNEL(2, 18), > > > + AD7606_SW_CHANNEL(3, 18), > > > + AD7606_SW_CHANNEL(4, 18), > > > + AD7606_SW_CHANNEL(5, 18), > > > + AD7606_SW_CHANNEL(6, 18), > > > + AD7606_SW_CHANNEL(7, 18), > > > +}; > > > + > > > static const unsigned int ad7606B_oversampling_avail[9] = { > > > 1, 2, 4, 8, 16, 32, 64, 128, 256 > > > }; > > > @@ -120,6 +132,19 @@ static int ad7606_spi_read_block(struct device *dev, > > > return 0; > > > } > > > > > > +static int ad7606_spi_read_block18to32(struct device *dev, > > > + int count, void *buf) > > > +{ > > > + struct spi_device *spi = to_spi_device(dev); > > > + struct spi_transfer xfer = { > > > + .bits_per_word = 18, > > > + .len = count, > > > > Isn't count the number of words? .len needs to be the number > > of bytes, so 4 * count. > > Oh. > Hmm, I need to check. > I assumed .len is the count of elements, not of bytes. > > > > > > + .rx_buf = buf, > > > + }; > > > + > > > + return spi_sync_transfer(spi, &xfer, 1); > > > +} > > > + > > > static int ad7606_spi_reg_read(struct ad7606_state *st, unsigned int addr) > > > { > > > struct spi_device *spi = to_spi_device(st->dev); > > > @@ -283,6 +308,19 @@ static int ad7606B_sw_mode_config(struct iio_dev *indio_dev) > > > return 0; > > > } > > > > > > +static int ad7606c_18_sw_mode_config(struct iio_dev *indio_dev) > > > +{ > > > + int ret; > > > + > > > + ret = ad7606B_sw_mode_config(indio_dev); > > > + if (ret) > > > + return ret; > > > + > > > + indio_dev->channels = ad7606c_18_sw_channels; > > > + > > > + return 0; > > > +} > > > + > > > static const struct ad7606_bus_ops ad7606_spi_bops = { > > > .read_block = ad7606_spi_read_block, > > > }; > > > @@ -305,6 +343,15 @@ static const struct ad7606_bus_ops ad7606B_spi_bops = { > > > .sw_mode_config = ad7606B_sw_mode_config, > > > }; > > > > > > +static const struct ad7606_bus_ops ad7606c_18_spi_bops = { > > > + .read_block = ad7606_spi_read_block18to32, > > > + .reg_read = ad7606_spi_reg_read, > > > + .reg_write = ad7606_spi_reg_write, > > > + .write_mask = ad7606_spi_write_mask, > > > + .rd_wr_cmd = ad7606B_spi_rd_wr_cmd, > > > + .sw_mode_config = ad7606c_18_sw_mode_config, > > > +}; > > > + > > > static int ad7606_spi_probe(struct spi_device *spi) > > > { > > > const struct spi_device_id *id = spi_get_device_id(spi); > > > @@ -315,8 +362,12 @@ static int ad7606_spi_probe(struct spi_device *spi) > > > bops = &ad7616_spi_bops; > > > break; > > > case ID_AD7606B: > > > + case ID_AD7606C_16: > > > bops = &ad7606B_spi_bops; > > > break; > > > + case ID_AD7606C_18: > > > + bops = &ad7606c_18_spi_bops; > > > + break; > > > default: > > > bops = &ad7606_spi_bops; > > > break; > > > @@ -333,6 +384,8 @@ static const struct spi_device_id ad7606_id_table[] = { > > > { "ad7606-6", ID_AD7606_6 }, > > > { "ad7606-8", ID_AD7606_8 }, > > > { "ad7606b", ID_AD7606B }, > > > + { "ad7606c-16", ID_AD7606C_16 }, > > > + { "ad7606c-18", ID_AD7606C_18 }, > > > { "ad7616", ID_AD7616 }, > > > { } > > > }; > > > @@ -344,6 +397,8 @@ static const struct of_device_id ad7606_of_match[] = { > > > { .compatible = "adi,ad7606-6" }, > > > { .compatible = "adi,ad7606-8" }, > > > { .compatible = "adi,ad7606b" }, > > > + { .compatible = "adi,ad7606c-16" }, > > > + { .compatible = "adi,ad7606c-18" }, > > > { .compatible = "adi,ad7616" }, > > > { } > > > }; > >
diff --git a/drivers/iio/adc/ad7606.c b/drivers/iio/adc/ad7606.c index 4c3fbb28f790..999c4411859e 100644 --- a/drivers/iio/adc/ad7606.c +++ b/drivers/iio/adc/ad7606.c @@ -28,14 +28,44 @@ #include "ad7606.h" +typedef void (*ad7606c_chan_setup_cb_t)(struct ad7606_state *st, int ch, + bool bipolar, bool differential); + /* * Scales are computed as 5000/32768 and 10000/32768 respectively, * so that when applied to the raw values they provide mV values */ -static const unsigned int ad7606_scale_avail[2] = { +static const unsigned int ad7606_16bit_hw_scale_avail[2] = { 152588, 305176 }; +static const unsigned int ad7606_18bit_hw_scale_avail[2] = { + 38147, 76294 +}; + +static const unsigned int ad7606c_16_scale_single_ended_unipolar_avail[3] = { + 76294, 152588, 190735, +}; + +static const unsigned int ad7606c_16_scale_single_ended_bipolar_avail[5] = { + 76294, 152588, 190735, 305176, 381470 +}; + +static const unsigned int ad7606c_16_scale_differential_bipolar_avail[4] = { + 152588, 305176, 381470, 610352 +}; + +static const unsigned int ad7606c_18_scale_single_ended_unipolar_avail[3] = { + 19073, 38147, 47684 +}; + +static const unsigned int ad7606c_18_scale_single_ended_bipolar_avail[5] = { + 19073, 38147, 47684, 76294, 95367 +}; + +static const unsigned int ad7606c_18_scale_differential_bipolar_avail[4] = { + 38147, 76294, 95367, 152588 +}; static const unsigned int ad7616_sw_scale_avail[3] = { 76293, 152588, 305176 @@ -82,11 +112,19 @@ static int ad7606_reg_access(struct iio_dev *indio_dev, } } -static int ad7606_read_samples(struct ad7606_state *st) +static int ad7606_read_samples(struct ad7606_state *st, bool sign_extend_samples) { + unsigned int storagebits = st->chip_info->channels[1].scan_type.storagebits; unsigned int num = st->chip_info->num_channels - 1; - u16 *data = st->data; - int ret; + u32 *data32 = st->data.d32; + u16 *data16 = st->data.d16; + void *data; + int i, ret; + + if (storagebits > 16) + data = data32; + else + data = data16; /* * The frstdata signal is set to high while and after reading the sample @@ -108,11 +146,25 @@ static int ad7606_read_samples(struct ad7606_state *st) return -EIO; } - data++; + if (storagebits > 16) + data32++; + else + data16++; num--; } - return st->bops->read_block(st->dev, num, data); + ret = st->bops->read_block(st->dev, num, data); + if (ret) + return ret; + + if (storagebits == 16 || !sign_extend_samples) + return 0; + + /* For 18 bit samples, we need to sign-extend samples to 32 bits */ + for (i = 0; i < num; i++) + data32[i] = sign_extend32(data32[i], 17); + + return 0; } static irqreturn_t ad7606_trigger_handler(int irq, void *p) @@ -124,11 +176,11 @@ static irqreturn_t ad7606_trigger_handler(int irq, void *p) guard(mutex)(&st->lock); - ret = ad7606_read_samples(st); + ret = ad7606_read_samples(st, true); if (ret) goto error_ret; - iio_push_to_buffers_with_timestamp(indio_dev, st->data, + iio_push_to_buffers_with_timestamp(indio_dev, st->data.d16, iio_get_time_ns(indio_dev)); error_ret: iio_trigger_notify_done(indio_dev->trig); @@ -142,6 +194,7 @@ static int ad7606_scan_direct(struct iio_dev *indio_dev, unsigned int ch, int *val) { struct ad7606_state *st = iio_priv(indio_dev); + unsigned int storagebits = st->chip_info->channels[1].scan_type.storagebits; int ret; gpiod_set_value(st->gpio_convst, 1); @@ -152,9 +205,13 @@ static int ad7606_scan_direct(struct iio_dev *indio_dev, unsigned int ch, goto error_ret; } - ret = ad7606_read_samples(st); - if (ret == 0) - *val = sign_extend32(st->data[ch], 15); + ret = ad7606_read_samples(st, false); + if (ret == 0) { + if (storagebits > 16) + *val = sign_extend32(st->data.d32[ch], 17); + else + *val = sign_extend32(st->data.d16[ch], 15); + } error_ret: gpiod_set_value(st->gpio_convst, 0); @@ -267,7 +324,7 @@ static int ad7606_write_raw(struct iio_dev *indio_dev, ch = chan->address; cs = &st->chan_scales[ch]; i = find_closest(val2, cs->scale_avail, cs->num_scales); - ret = st->write_scale(indio_dev, ch, i); + ret = st->write_scale(indio_dev, ch, i + cs->reg_offset); if (ret < 0) return ret; cs->range = i; @@ -350,6 +407,18 @@ static const struct iio_chan_spec ad7606_channels_16bit[] = { AD7606_CHANNEL(7, 16), }; +static const struct iio_chan_spec ad7606_channels_18bit[] = { + IIO_CHAN_SOFT_TIMESTAMP(8), + AD7606_CHANNEL(0, 18), + AD7606_CHANNEL(1, 18), + AD7606_CHANNEL(2, 18), + AD7606_CHANNEL(3, 18), + AD7606_CHANNEL(4, 18), + AD7606_CHANNEL(5, 18), + AD7606_CHANNEL(6, 18), + AD7606_CHANNEL(7, 18), +}; + /* * The current assumption that this driver makes for AD7616, is that it's * working in Hardware Mode with Serial, Burst and Sequencer modes activated. @@ -410,6 +479,18 @@ static const struct ad7606_chip_info ad7606_chip_info_tbl[] = { .oversampling_avail = ad7606_oversampling_avail, .oversampling_num = ARRAY_SIZE(ad7606_oversampling_avail), }, + [ID_AD7606C_16] = { + .channels = ad7606_channels_16bit, + .num_channels = 9, + .oversampling_avail = ad7606_oversampling_avail, + .oversampling_num = ARRAY_SIZE(ad7606_oversampling_avail), + }, + [ID_AD7606C_18] = { + .channels = ad7606_channels_18bit, + .num_channels = 9, + .oversampling_avail = ad7606_oversampling_avail, + .oversampling_num = ARRAY_SIZE(ad7606_oversampling_avail), + }, [ID_AD7616] = { .channels = ad7616_channels, .num_channels = 17, @@ -581,7 +662,122 @@ static const struct iio_trigger_ops ad7606_trigger_ops = { .validate_device = iio_trigger_validate_own_device, }; -static int ad7606_sw_mode_setup(struct iio_dev *indio_dev) +static void ad7606c_18_chan_setup(struct ad7606_state *st, int ch, + bool bipolar, bool differential) +{ + struct ad7606_chan_scale *cs = &st->chan_scales[ch]; + + if (differential) { + cs->scale_avail = + ad7606c_18_scale_differential_bipolar_avail; + cs->num_scales = + ARRAY_SIZE(ad7606c_18_scale_differential_bipolar_avail); + /* Bipolar differential ranges start at 8 (b1000) */ + cs->reg_offset = 8; + cs->range = 1; + } else if (bipolar) { + cs->scale_avail = + ad7606c_18_scale_single_ended_bipolar_avail; + cs->num_scales = + ARRAY_SIZE(ad7606c_18_scale_single_ended_bipolar_avail); + cs->range = 3; + } else { + cs->scale_avail = + ad7606c_18_scale_single_ended_unipolar_avail; + cs->num_scales = + ARRAY_SIZE(ad7606c_18_scale_single_ended_unipolar_avail); + /* Unipolar single-ended ranges start at 5 (b0101) */ + cs->reg_offset = 5; + cs->range = 1; + } +} + +static void ad7606c_16_chan_setup(struct ad7606_state *st, int ch, + bool bipolar, bool differential) +{ + struct ad7606_chan_scale *cs = &st->chan_scales[ch]; + + if (differential) { + cs->scale_avail = + ad7606c_16_scale_differential_bipolar_avail; + cs->num_scales = + ARRAY_SIZE(ad7606c_16_scale_differential_bipolar_avail); + /* Bipolar differential ranges start at 8 (b1000) */ + cs->reg_offset = 8; + cs->range = 1; + } else if (bipolar) { + cs->scale_avail = + ad7606c_16_scale_single_ended_bipolar_avail; + cs->num_scales = + ARRAY_SIZE(ad7606c_16_scale_single_ended_bipolar_avail); + cs->range = 3; + } else { + cs->scale_avail = + ad7606c_16_scale_single_ended_unipolar_avail; + cs->num_scales = + ARRAY_SIZE(ad7606c_16_scale_single_ended_unipolar_avail); + /* Unipolar single-ended ranges start at 5 (b0101) */ + cs->reg_offset = 5; + cs->range = 1; + } +} + +static int ad7606c_sw_mode_setup_channels(struct iio_dev *indio_dev, + ad7606c_chan_setup_cb_t chan_setup_cb) +{ + unsigned int num_channels = indio_dev->num_channels - 1; + struct ad7606_state *st = iio_priv(indio_dev); + bool chan_configured[AD760X_MAX_CHANNELS] = {}; + struct device *dev = st->dev; + int ret; + u32 ch; + + /* We need to hook this first */ + ret = st->bops->sw_mode_config(indio_dev); + if (ret) + return ret; + + device_for_each_child_node_scoped(dev, child) { + bool bipolar, differential; + u32 pins[2]; + + ret = fwnode_property_read_u32(child, "reg", &ch); + if (ret) + continue; + + /* channel number (here) is from 1 to num_channels */ + if (ch == 0 || ch > num_channels) { + dev_warn(st->dev, + "Invalid channel number (ignoring): %d\n", ch); + continue; + } + + bipolar = fwnode_property_present(child, "bipolar"); + + ret = fwnode_property_read_u32_array(child, "diff-channels", + pins, ARRAY_SIZE(pins)); + /* Channel is differential, if pins are the same as 'reg' */ + if (ret == 0 && pins[0] == ch && pins[1] == ch) + differential = true; + else + differential = false; + + ch--; + + chan_setup_cb(st, ch, bipolar, differential); + chan_configured[ch] = true; + } + + /* Apply default configuration to unconfigured (via DT) channels */ + for (ch = 0; ch < num_channels; ch++) { + if (!chan_configured[ch]) + chan_setup_cb(st, ch, false, false); + } + + return 0; +} + +static int ad7606_sw_mode_setup(struct iio_dev *indio_dev, unsigned int id) { unsigned int num_channels = indio_dev->num_channels - 1; struct ad7606_state *st = iio_priv(indio_dev); @@ -596,17 +792,30 @@ static int ad7606_sw_mode_setup(struct iio_dev *indio_dev) indio_dev->info = &ad7606_info_sw_mode; - /* Scale of 0.076293 is only available in sw mode */ - /* After reset, in software mode, ±10 V is set by default */ - for (ch = 0; ch < num_channels; ch++) { - struct ad7606_chan_scale *cs = &st->chan_scales[ch]; + switch (id) { + case ID_AD7606C_18: + ret = ad7606c_sw_mode_setup_channels(indio_dev, + ad7606c_18_chan_setup); + break; + case ID_AD7606C_16: + ret = ad7606c_sw_mode_setup_channels(indio_dev, + ad7606c_16_chan_setup); + break; + default: + /* Scale of 0.076293 is only available in sw mode */ + /* After reset, in software mode, ±10 V is set by default */ + for (ch = 0; ch < num_channels; ch++) { + struct ad7606_chan_scale *cs = &st->chan_scales[ch]; + + cs->scale_avail = ad7616_sw_scale_avail; + cs->num_scales = ARRAY_SIZE(ad7616_sw_scale_avail); + cs->range = 2; + } - cs->scale_avail = ad7616_sw_scale_avail; - cs->num_scales = ARRAY_SIZE(ad7616_sw_scale_avail); - cs->range = 2; + ret = st->bops->sw_mode_config(indio_dev); + break; } - ret = st->bops->sw_mode_config(indio_dev); if (ret) return ret; @@ -655,9 +864,16 @@ int ad7606_probe(struct device *dev, int irq, void __iomem *base_address, st->oversampling = 1; cs = &st->chan_scales[0]; - cs->range = 0; - cs->scale_avail = ad7606_scale_avail; - cs->num_scales = ARRAY_SIZE(ad7606_scale_avail); + switch (id) { + case ID_AD7606C_18: + cs->scale_avail = ad7606_18bit_hw_scale_avail; + cs->num_scales = ARRAY_SIZE(ad7606_18bit_hw_scale_avail); + break; + default: + cs->scale_avail = ad7606_16bit_hw_scale_avail; + cs->num_scales = ARRAY_SIZE(ad7606_16bit_hw_scale_avail); + break; + } ret = devm_regulator_get_enable(dev, "avcc"); if (ret) @@ -706,7 +922,7 @@ int ad7606_probe(struct device *dev, int irq, void __iomem *base_address, st->write_scale = ad7606_write_scale_hw; st->write_os = ad7606_write_os_hw; - ret = ad7606_sw_mode_setup(indio_dev); + ret = ad7606_sw_mode_setup(indio_dev, id); if (ret) return ret; diff --git a/drivers/iio/adc/ad7606.h b/drivers/iio/adc/ad7606.h index 2113ad460c0f..6b0897aa2dc7 100644 --- a/drivers/iio/adc/ad7606.h +++ b/drivers/iio/adc/ad7606.h @@ -22,7 +22,7 @@ .scan_type = { \ .sign = 's', \ .realbits = (bits), \ - .storagebits = (bits), \ + .storagebits = (bits) > 16 ? 32 : 16, \ .endianness = IIO_CPU, \ }, \ } @@ -45,7 +45,7 @@ .scan_type = { \ .sign = 's', \ .realbits = (bits), \ - .storagebits = (bits), \ + .storagebits = (bits) > 16 ? 32 : 16, \ .endianness = IIO_CPU, \ }, \ } @@ -88,6 +88,8 @@ struct ad7606_chip_info { * such that it can be read via the 'read_avail' hook * @num_scales number of elements stored in the scale_avail array * @range voltage range selection, selects which scale to apply + * @reg_offset offset for the register value, to be applied when + * writing the value of 'range' to the register value */ struct ad7606_chan_scale { #define AD760X_MAX_SCALE_SHOW (AD760X_MAX_CHANNELS * 2) @@ -95,6 +97,7 @@ struct ad7606_chan_scale { int scale_avail_show[AD760X_MAX_SCALE_SHOW]; unsigned int num_scales; unsigned int range; + unsigned int reg_offset; }; /** @@ -151,9 +154,13 @@ struct ad7606_state { /* * DMA (thus cache coherency maintenance) may require the * transfer buffers to live in their own cache lines. - * 16 * 16-bit samples + 64-bit timestamp + * 16 * 16-bit samples + 64-bit timestamp - for AD7616 + * 8 * 32-bit samples + 64-bit timestamp - for AD7616C-18 (and similar) */ - unsigned short data[20] __aligned(IIO_DMA_MINALIGN); + union { + unsigned short d16[20]; + unsigned int d32[10]; + } data __aligned(IIO_DMA_MINALIGN); __be16 d16[2]; }; @@ -192,6 +199,8 @@ enum ad7606_supported_device_ids { ID_AD7606_6, ID_AD7606_4, ID_AD7606B, + ID_AD7606C_16, + ID_AD7606C_18, ID_AD7616, }; diff --git a/drivers/iio/adc/ad7606_spi.c b/drivers/iio/adc/ad7606_spi.c index e00f58a6a0e9..b8d630ad156d 100644 --- a/drivers/iio/adc/ad7606_spi.c +++ b/drivers/iio/adc/ad7606_spi.c @@ -77,6 +77,18 @@ static const struct iio_chan_spec ad7606b_sw_channels[] = { AD7606_SW_CHANNEL(7, 16), }; +static const struct iio_chan_spec ad7606c_18_sw_channels[] = { + IIO_CHAN_SOFT_TIMESTAMP(8), + AD7606_SW_CHANNEL(0, 18), + AD7606_SW_CHANNEL(1, 18), + AD7606_SW_CHANNEL(2, 18), + AD7606_SW_CHANNEL(3, 18), + AD7606_SW_CHANNEL(4, 18), + AD7606_SW_CHANNEL(5, 18), + AD7606_SW_CHANNEL(6, 18), + AD7606_SW_CHANNEL(7, 18), +}; + static const unsigned int ad7606B_oversampling_avail[9] = { 1, 2, 4, 8, 16, 32, 64, 128, 256 }; @@ -120,6 +132,19 @@ static int ad7606_spi_read_block(struct device *dev, return 0; } +static int ad7606_spi_read_block18to32(struct device *dev, + int count, void *buf) +{ + struct spi_device *spi = to_spi_device(dev); + struct spi_transfer xfer = { + .bits_per_word = 18, + .len = count, + .rx_buf = buf, + }; + + return spi_sync_transfer(spi, &xfer, 1); +} + static int ad7606_spi_reg_read(struct ad7606_state *st, unsigned int addr) { struct spi_device *spi = to_spi_device(st->dev); @@ -283,6 +308,19 @@ static int ad7606B_sw_mode_config(struct iio_dev *indio_dev) return 0; } +static int ad7606c_18_sw_mode_config(struct iio_dev *indio_dev) +{ + int ret; + + ret = ad7606B_sw_mode_config(indio_dev); + if (ret) + return ret; + + indio_dev->channels = ad7606c_18_sw_channels; + + return 0; +} + static const struct ad7606_bus_ops ad7606_spi_bops = { .read_block = ad7606_spi_read_block, }; @@ -305,6 +343,15 @@ static const struct ad7606_bus_ops ad7606B_spi_bops = { .sw_mode_config = ad7606B_sw_mode_config, }; +static const struct ad7606_bus_ops ad7606c_18_spi_bops = { + .read_block = ad7606_spi_read_block18to32, + .reg_read = ad7606_spi_reg_read, + .reg_write = ad7606_spi_reg_write, + .write_mask = ad7606_spi_write_mask, + .rd_wr_cmd = ad7606B_spi_rd_wr_cmd, + .sw_mode_config = ad7606c_18_sw_mode_config, +}; + static int ad7606_spi_probe(struct spi_device *spi) { const struct spi_device_id *id = spi_get_device_id(spi); @@ -315,8 +362,12 @@ static int ad7606_spi_probe(struct spi_device *spi) bops = &ad7616_spi_bops; break; case ID_AD7606B: + case ID_AD7606C_16: bops = &ad7606B_spi_bops; break; + case ID_AD7606C_18: + bops = &ad7606c_18_spi_bops; + break; default: bops = &ad7606_spi_bops; break; @@ -333,6 +384,8 @@ static const struct spi_device_id ad7606_id_table[] = { { "ad7606-6", ID_AD7606_6 }, { "ad7606-8", ID_AD7606_8 }, { "ad7606b", ID_AD7606B }, + { "ad7606c-16", ID_AD7606C_16 }, + { "ad7606c-18", ID_AD7606C_18 }, { "ad7616", ID_AD7616 }, { } }; @@ -344,6 +397,8 @@ static const struct of_device_id ad7606_of_match[] = { { .compatible = "adi,ad7606-6" }, { .compatible = "adi,ad7606-8" }, { .compatible = "adi,ad7606b" }, + { .compatible = "adi,ad7606c-16" }, + { .compatible = "adi,ad7606c-18" }, { .compatible = "adi,ad7616" }, { } };
The AD7606C-16 and AD7606C-18 are pretty similar with the AD7606B. The main difference between AD7606C-16 & AD7606C-18 is the precision in bits (16 vs 18). Because of that, some scales need to be defined for the 18-bit variants, as they need to be computed against 2**18 (vs 2**16 for the 16 bit-variants). Because the AD7606C-16,18 also supports bipolar & differential channels, for SW-mode, the default range of 10 V or ±10V should be set at probe. On reset, the default range (in the registers) is set to value 0x3 which corresponds to '±10 V single-ended range', regardless of bipolar or differential configuration. Aside from the scale/ranges, the AD7606C-16 is similar to the AD7606B. The AD7606C-18 variant offers 18-bit precision. Because of this, the requirement to use this chip is that the SPI controller supports padding of 18-bit sequences to 32-bit arrays. Datasheet links: https://www.analog.com/media/en/technical-documentation/data-sheets/ad7606c-16.pdf https://www.analog.com/media/en/technical-documentation/data-sheets/ad7606c-18.pdf Signed-off-by: Alexandru Ardelean <aardelean@baylibre.com> --- drivers/iio/adc/ad7606.c | 266 +++++++++++++++++++++++++++++++---- drivers/iio/adc/ad7606.h | 17 ++- drivers/iio/adc/ad7606_spi.c | 55 ++++++++ 3 files changed, 309 insertions(+), 29 deletions(-)