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: https://www.analog.com/media/en/technical-documentation/data-sheets/ad7606c-16.pdf
Datasheet: 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 | 263 +++++++++++++++++++++++++++++++++--
drivers/iio/adc/ad7606.h | 16 ++-
drivers/iio/adc/ad7606_spi.c | 55 ++++++++
3 files changed, 322 insertions(+), 12 deletions(-)
diff --git a/drivers/iio/adc/ad7606.c b/drivers/iio/adc/ad7606.c
index b909ee14fd81..f04e5660d2f8 100644
--- a/drivers/iio/adc/ad7606.c
+++ b/drivers/iio/adc/ad7606.c
@@ -36,6 +36,33 @@ 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_16bit_single_ended_unipolar_scale_avail[3] = {
+ 76294, 152588, 190735,
+};
+
+static const unsigned int ad7606c_16bit_single_ended_bipolar_scale_avail[5] = {
+ 76294, 152588, 190735, 305176, 381470
+};
+
+static const unsigned int ad7606c_16bit_differential_bipolar_scale_avail[4] = {
+ 152588, 305176, 381470, 610352
+};
+
+static const unsigned int ad7606c_18bit_single_ended_unipolar_scale_avail[3] = {
+ 19073, 38147, 47684
+};
+
+static const unsigned int ad7606c_18bit_single_ended_bipolar_scale_avail[5] = {
+ 19073, 38147, 47684, 76294, 95367
+};
+
+static const unsigned int ad7606c_18bit_differential_bipolar_scale_avail[4] = {
+ 38147, 76294, 95367, 152588
+};
static const unsigned int ad7606_16bit_sw_scale_avail[3] = {
76293, 152588, 305176
@@ -62,7 +89,8 @@ int ad7606_reset(struct ad7606_state *st)
}
EXPORT_SYMBOL_NS_GPL(ad7606_reset, IIO_AD7606);
-static int ad7606_16bit_chan_scale_setup(struct ad7606_state *st, int ch)
+static int ad7606_16bit_chan_scale_setup(struct ad7606_state *st,
+ struct iio_chan_spec *chan, int ch)
{
struct ad7606_chan_scale *cs = &st->chan_scales[ch];
@@ -83,6 +111,173 @@ static int ad7606_16bit_chan_scale_setup(struct ad7606_state *st, int ch)
return 0;
}
+static int ad7606_get_chan_config(struct ad7606_state *st, int ch,
+ bool *bipolar, bool *differential)
+{
+ unsigned int num_channels = st->chip_info->num_channels - 1;
+ struct device *dev = st->dev;
+ int ret;
+
+ *bipolar = false;
+ *differential = false;
+
+ device_for_each_child_node_scoped(dev, child) {
+ u32 pins[2];
+ int reg;
+
+ ret = fwnode_property_read_u32(child, "reg", ®);
+ if (ret)
+ continue;
+
+ /* channel number (here) is from 1 to num_channels */
+ if (reg == 0 || reg > num_channels) {
+ dev_warn(dev,
+ "Invalid channel number (ignoring): %d\n", reg);
+ continue;
+ }
+
+ if (reg != (ch + 1))
+ continue;
+
+ *bipolar = fwnode_property_read_bool(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] != reg || pins[1] != reg)) {
+ dev_err(dev,
+ "Differential pins must be the same as 'reg'");
+ return -EINVAL;
+ }
+
+ *differential = (ret == 0);
+
+ if (*differential && !*bipolar) {
+ dev_err(dev,
+ "'bipolar' must be added for diff channel %d\n",
+ reg);
+ return -EINVAL;
+ }
+
+ return 0;
+ }
+
+ return 0;
+}
+
+static int ad7606c_18bit_chan_scale_setup(struct ad7606_state *st,
+ struct iio_chan_spec *chan, int ch)
+{
+ struct ad7606_chan_scale *cs = &st->chan_scales[ch];
+ bool bipolar, differential;
+ int ret;
+
+ if (!st->sw_mode_en) {
+ cs->range = 0;
+ cs->scale_avail = ad7606_18bit_hw_scale_avail;
+ cs->num_scales = ARRAY_SIZE(ad7606_18bit_hw_scale_avail);
+ return 0;
+ }
+
+ ret = ad7606_get_chan_config(st, ch, &bipolar, &differential);
+ if (ret)
+ return ret;
+
+ if (differential) {
+ cs->scale_avail = ad7606c_18bit_differential_bipolar_scale_avail;
+ cs->num_scales =
+ ARRAY_SIZE(ad7606c_18bit_differential_bipolar_scale_avail);
+ /* Bipolar differential ranges start at 8 (b1000) */
+ cs->reg_offset = 8;
+ cs->range = 1;
+ chan->differential = 1;
+ chan->channel2 = chan->channel;
+
+ return 0;
+ }
+
+ chan->differential = 0;
+
+ if (bipolar) {
+ cs->scale_avail = ad7606c_18bit_single_ended_bipolar_scale_avail;
+ cs->num_scales =
+ ARRAY_SIZE(ad7606c_18bit_single_ended_bipolar_scale_avail);
+ /* Bipolar single-ended ranges start at 0 (b0000) */
+ cs->reg_offset = 0;
+ cs->range = 3;
+ chan->scan_type.sign = 's';
+
+ return 0;
+ }
+
+ cs->scale_avail = ad7606c_18bit_single_ended_unipolar_scale_avail;
+ cs->num_scales =
+ ARRAY_SIZE(ad7606c_18bit_single_ended_unipolar_scale_avail);
+ /* Unipolar single-ended ranges start at 5 (b0101) */
+ cs->reg_offset = 5;
+ cs->range = 1;
+ chan->scan_type.sign = 'u';
+
+ return 0;
+}
+
+static int ad7606c_16bit_chan_scale_setup(struct ad7606_state *st,
+ struct iio_chan_spec *chan, int ch)
+{
+ struct ad7606_chan_scale *cs = &st->chan_scales[ch];
+ bool bipolar, differential;
+ int ret;
+
+ if (!st->sw_mode_en) {
+ cs->range = 0;
+ cs->scale_avail = ad7606_16bit_hw_scale_avail;
+ cs->num_scales = ARRAY_SIZE(ad7606_16bit_hw_scale_avail);
+ return 0;
+ }
+
+ ret = ad7606_get_chan_config(st, ch, &bipolar, &differential);
+ if (ret)
+ return ret;
+
+ if (differential) {
+ cs->scale_avail = ad7606c_16bit_differential_bipolar_scale_avail;
+ cs->num_scales =
+ ARRAY_SIZE(ad7606c_16bit_differential_bipolar_scale_avail);
+ /* Bipolar differential ranges start at 8 (b1000) */
+ cs->reg_offset = 8;
+ cs->range = 1;
+ chan->differential = 1;
+ chan->channel2 = chan->channel;
+ chan->scan_type.sign = 's';
+
+ return 0;
+ }
+
+ chan->differential = 0;
+
+ if (bipolar) {
+ cs->scale_avail = ad7606c_16bit_single_ended_bipolar_scale_avail;
+ cs->num_scales =
+ ARRAY_SIZE(ad7606c_16bit_single_ended_bipolar_scale_avail);
+ /* Bipolar single-ended ranges start at 0 (b0000) */
+ cs->reg_offset = 0;
+ cs->range = 3;
+ chan->scan_type.sign = 's';
+
+ return 0;
+ }
+
+ cs->scale_avail = ad7606c_16bit_single_ended_unipolar_scale_avail;
+ cs->num_scales =
+ ARRAY_SIZE(ad7606c_16bit_single_ended_unipolar_scale_avail);
+ /* Unipolar single-ended ranges start at 5 (b0101) */
+ cs->reg_offset = 5;
+ cs->range = 1;
+ chan->scan_type.sign = 'u';
+
+ return 0;
+}
+
static int ad7606_reg_access(struct iio_dev *indio_dev,
unsigned int reg,
unsigned int writeval,
@@ -107,9 +302,8 @@ static int ad7606_reg_access(struct iio_dev *indio_dev,
static int ad7606_read_samples(struct ad7606_state *st)
{
unsigned int num = st->chip_info->num_channels - 1;
- u16 *data = st->data;
- return st->bops->read_block(st->dev, num, data);
+ return st->bops->read_block(st->dev, num, &st->data);
}
static irqreturn_t ad7606_trigger_handler(int irq, void *p)
@@ -125,7 +319,7 @@ static irqreturn_t ad7606_trigger_handler(int irq, void *p)
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,
iio_get_time_ns(indio_dev));
error_ret:
iio_trigger_notify_done(indio_dev->trig);
@@ -139,6 +333,8 @@ 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;
+ const struct iio_chan_spec *chan;
int ret;
gpiod_set_value(st->gpio_convst, 1);
@@ -153,7 +349,19 @@ static int ad7606_scan_direct(struct iio_dev *indio_dev, unsigned int ch,
if (ret)
goto error_ret;
- *val = sign_extend32(st->data[ch], 15);
+ chan = &indio_dev->channels[ch + 1];
+ if (chan->scan_type.sign == 'u') {
+ if (storagebits > 16)
+ *val = st->data.buf32[ch];
+ else
+ *val = st->data.buf16[ch];
+ return 0;
+ }
+
+ if (storagebits > 16)
+ *val = sign_extend32(st->data.buf32[ch], 17);
+ else
+ *val = sign_extend32(st->data.buf16[ch], 15);
error_ret:
gpiod_set_value(st->gpio_convst, 0);
@@ -266,7 +474,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;
@@ -349,6 +557,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.
@@ -414,6 +634,20 @@ 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,
+ .scale_setup_cb = ad7606c_16bit_chan_scale_setup,
+ .oversampling_avail = ad7606_oversampling_avail,
+ .oversampling_num = ARRAY_SIZE(ad7606_oversampling_avail),
+ },
+ [ID_AD7606C_18] = {
+ .channels = ad7606_channels_18bit,
+ .num_channels = 9,
+ .scale_setup_cb = ad7606c_18bit_chan_scale_setup,
+ .oversampling_avail = ad7606_oversampling_avail,
+ .oversampling_num = ARRAY_SIZE(ad7606_oversampling_avail),
+ },
[ID_AD7616] = {
.channels = ad7616_channels,
.num_channels = 17,
@@ -586,7 +820,7 @@ 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 int ad7606_sw_mode_setup(struct iio_dev *indio_dev, unsigned int id)
{
struct ad7606_state *st = iio_priv(indio_dev);
@@ -604,13 +838,24 @@ static int ad7606_chan_scales_setup(struct iio_dev *indio_dev)
{
unsigned int num_channels = indio_dev->num_channels - 1;
struct ad7606_state *st = iio_priv(indio_dev);
+ struct iio_chan_spec *chans;
+ size_t size;
int ch, ret;
+ /* Clone IIO channels, since some may be differential */
+ size = indio_dev->num_channels * sizeof(*indio_dev->channels);
+ chans = devm_kzalloc(st->dev, size, GFP_KERNEL);
+ if (!chans)
+ return -ENOMEM;
+
+ memcpy(chans, indio_dev->channels, size);
+ indio_dev->channels = chans;
+
for (ch = 0; ch < num_channels; ch++) {
struct ad7606_chan_scale *cs;
int i;
- ret = st->chip_info->scale_setup_cb(st, ch);
+ ret = st->chip_info->scale_setup_cb(st, &chans[ch + 1], ch);
if (ret)
return ret;
@@ -698,7 +943,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 25e84efd15c3..14ee75aa225b 100644
--- a/drivers/iio/adc/ad7606.h
+++ b/drivers/iio/adc/ad7606.h
@@ -63,7 +63,8 @@
struct ad7606_state;
-typedef int (*ad7606_scale_setup_cb_t)(struct ad7606_state *st, int ch);
+typedef int (*ad7606_scale_setup_cb_t)(struct ad7606_state *st,
+ struct iio_chan_spec *chan, int ch);
/**
* struct ad7606_chip_info - chip specific information
@@ -94,6 +95,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_SCALES 16
@@ -102,6 +105,7 @@ struct ad7606_chan_scale {
int scale_avail_show[AD760X_MAX_SCALE_SHOW];
unsigned int num_scales;
unsigned int range;
+ unsigned int reg_offset;
};
/**
@@ -158,9 +162,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 {
+ u16 buf16[20];
+ u32 buf32[10];
+ } data __aligned(IIO_DMA_MINALIGN);
__be16 d16[2];
};
@@ -201,6 +209,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..143440e73aab 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 * sizeof(u32),
+ .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" },
{ }
};
--
2.46.0
On Thu, Sep 19, 2024 at 4:05 PM Alexandru Ardelean <aardelean@baylibre.com> 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: https://www.analog.com/media/en/technical-documentation/data-sheets/ad7606c-16.pdf > Datasheet: 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 | 263 +++++++++++++++++++++++++++++++++-- > drivers/iio/adc/ad7606.h | 16 ++- > drivers/iio/adc/ad7606_spi.c | 55 ++++++++ > 3 files changed, 322 insertions(+), 12 deletions(-) > > diff --git a/drivers/iio/adc/ad7606.c b/drivers/iio/adc/ad7606.c > index b909ee14fd81..f04e5660d2f8 100644 > --- a/drivers/iio/adc/ad7606.c > +++ b/drivers/iio/adc/ad7606.c > @@ -36,6 +36,33 @@ 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_16bit_single_ended_unipolar_scale_avail[3] = { > + 76294, 152588, 190735, > +}; > + > +static const unsigned int ad7606c_16bit_single_ended_bipolar_scale_avail[5] = { > + 76294, 152588, 190735, 305176, 381470 > +}; > + > +static const unsigned int ad7606c_16bit_differential_bipolar_scale_avail[4] = { > + 152588, 305176, 381470, 610352 > +}; > + > +static const unsigned int ad7606c_18bit_single_ended_unipolar_scale_avail[3] = { > + 19073, 38147, 47684 > +}; > + > +static const unsigned int ad7606c_18bit_single_ended_bipolar_scale_avail[5] = { > + 19073, 38147, 47684, 76294, 95367 > +}; > + > +static const unsigned int ad7606c_18bit_differential_bipolar_scale_avail[4] = { > + 38147, 76294, 95367, 152588 > +}; > > static const unsigned int ad7606_16bit_sw_scale_avail[3] = { > 76293, 152588, 305176 > @@ -62,7 +89,8 @@ int ad7606_reset(struct ad7606_state *st) > } > EXPORT_SYMBOL_NS_GPL(ad7606_reset, IIO_AD7606); > > -static int ad7606_16bit_chan_scale_setup(struct ad7606_state *st, int ch) > +static int ad7606_16bit_chan_scale_setup(struct ad7606_state *st, > + struct iio_chan_spec *chan, int ch) > { > struct ad7606_chan_scale *cs = &st->chan_scales[ch]; > > @@ -83,6 +111,173 @@ static int ad7606_16bit_chan_scale_setup(struct ad7606_state *st, int ch) > return 0; > } > > +static int ad7606_get_chan_config(struct ad7606_state *st, int ch, > + bool *bipolar, bool *differential) > +{ > + unsigned int num_channels = st->chip_info->num_channels - 1; > + struct device *dev = st->dev; > + int ret; > + > + *bipolar = false; > + *differential = false; > + > + device_for_each_child_node_scoped(dev, child) { > + u32 pins[2]; > + int reg; > + > + ret = fwnode_property_read_u32(child, "reg", ®); > + if (ret) > + continue; > + > + /* channel number (here) is from 1 to num_channels */ > + if (reg == 0 || reg > num_channels) { > + dev_warn(dev, > + "Invalid channel number (ignoring): %d\n", reg); > + continue; > + } > + > + if (reg != (ch + 1)) > + continue; > + > + *bipolar = fwnode_property_read_bool(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] != reg || pins[1] != reg)) { > + dev_err(dev, > + "Differential pins must be the same as 'reg'"); > + return -EINVAL; > + } > + > + *differential = (ret == 0); > + > + if (*differential && !*bipolar) { > + dev_err(dev, > + "'bipolar' must be added for diff channel %d\n", > + reg); > + return -EINVAL; > + } > + > + return 0; > + } > + > + return 0; > +} > + > +static int ad7606c_18bit_chan_scale_setup(struct ad7606_state *st, > + struct iio_chan_spec *chan, int ch) > +{ > + struct ad7606_chan_scale *cs = &st->chan_scales[ch]; > + bool bipolar, differential; > + int ret; > + > + if (!st->sw_mode_en) { > + cs->range = 0; > + cs->scale_avail = ad7606_18bit_hw_scale_avail; > + cs->num_scales = ARRAY_SIZE(ad7606_18bit_hw_scale_avail); > + return 0; > + } > + > + ret = ad7606_get_chan_config(st, ch, &bipolar, &differential); > + if (ret) > + return ret; > + > + if (differential) { > + cs->scale_avail = ad7606c_18bit_differential_bipolar_scale_avail; > + cs->num_scales = > + ARRAY_SIZE(ad7606c_18bit_differential_bipolar_scale_avail); > + /* Bipolar differential ranges start at 8 (b1000) */ > + cs->reg_offset = 8; > + cs->range = 1; > + chan->differential = 1; > + chan->channel2 = chan->channel; > + > + return 0; > + } > + > + chan->differential = 0; > + > + if (bipolar) { > + cs->scale_avail = ad7606c_18bit_single_ended_bipolar_scale_avail; > + cs->num_scales = > + ARRAY_SIZE(ad7606c_18bit_single_ended_bipolar_scale_avail); > + /* Bipolar single-ended ranges start at 0 (b0000) */ > + cs->reg_offset = 0; > + cs->range = 3; > + chan->scan_type.sign = 's'; > + > + return 0; > + } > + > + cs->scale_avail = ad7606c_18bit_single_ended_unipolar_scale_avail; > + cs->num_scales = > + ARRAY_SIZE(ad7606c_18bit_single_ended_unipolar_scale_avail); > + /* Unipolar single-ended ranges start at 5 (b0101) */ > + cs->reg_offset = 5; > + cs->range = 1; > + chan->scan_type.sign = 'u'; > + > + return 0; > +} > + > +static int ad7606c_16bit_chan_scale_setup(struct ad7606_state *st, > + struct iio_chan_spec *chan, int ch) > +{ > + struct ad7606_chan_scale *cs = &st->chan_scales[ch]; > + bool bipolar, differential; > + int ret; > + > + if (!st->sw_mode_en) { > + cs->range = 0; > + cs->scale_avail = ad7606_16bit_hw_scale_avail; > + cs->num_scales = ARRAY_SIZE(ad7606_16bit_hw_scale_avail); > + return 0; > + } > + > + ret = ad7606_get_chan_config(st, ch, &bipolar, &differential); > + if (ret) > + return ret; > + > + if (differential) { > + cs->scale_avail = ad7606c_16bit_differential_bipolar_scale_avail; > + cs->num_scales = > + ARRAY_SIZE(ad7606c_16bit_differential_bipolar_scale_avail); > + /* Bipolar differential ranges start at 8 (b1000) */ > + cs->reg_offset = 8; > + cs->range = 1; > + chan->differential = 1; > + chan->channel2 = chan->channel; > + chan->scan_type.sign = 's'; > + > + return 0; > + } > + > + chan->differential = 0; > + > + if (bipolar) { > + cs->scale_avail = ad7606c_16bit_single_ended_bipolar_scale_avail; > + cs->num_scales = > + ARRAY_SIZE(ad7606c_16bit_single_ended_bipolar_scale_avail); > + /* Bipolar single-ended ranges start at 0 (b0000) */ > + cs->reg_offset = 0; > + cs->range = 3; > + chan->scan_type.sign = 's'; > + > + return 0; > + } > + > + cs->scale_avail = ad7606c_16bit_single_ended_unipolar_scale_avail; > + cs->num_scales = > + ARRAY_SIZE(ad7606c_16bit_single_ended_unipolar_scale_avail); > + /* Unipolar single-ended ranges start at 5 (b0101) */ > + cs->reg_offset = 5; > + cs->range = 1; > + chan->scan_type.sign = 'u'; > + > + return 0; > +} > + > static int ad7606_reg_access(struct iio_dev *indio_dev, > unsigned int reg, > unsigned int writeval, > @@ -107,9 +302,8 @@ static int ad7606_reg_access(struct iio_dev *indio_dev, > static int ad7606_read_samples(struct ad7606_state *st) > { > unsigned int num = st->chip_info->num_channels - 1; > - u16 *data = st->data; > > - return st->bops->read_block(st->dev, num, data); > + return st->bops->read_block(st->dev, num, &st->data); > } > > static irqreturn_t ad7606_trigger_handler(int irq, void *p) > @@ -125,7 +319,7 @@ static irqreturn_t ad7606_trigger_handler(int irq, void *p) > 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, > iio_get_time_ns(indio_dev)); > error_ret: > iio_trigger_notify_done(indio_dev->trig); > @@ -139,6 +333,8 @@ 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; > + const struct iio_chan_spec *chan; > int ret; > > gpiod_set_value(st->gpio_convst, 1); > @@ -153,7 +349,19 @@ static int ad7606_scan_direct(struct iio_dev *indio_dev, unsigned int ch, > if (ret) > goto error_ret; > > - *val = sign_extend32(st->data[ch], 15); > + chan = &indio_dev->channels[ch + 1]; > + if (chan->scan_type.sign == 'u') { > + if (storagebits > 16) > + *val = st->data.buf32[ch]; > + else > + *val = st->data.buf16[ch]; > + return 0; Arrggh... I messed up here. Guillaume found a bug here, where this should be "goto error_ret" or do an "if () {} else {}" How should we do it here? Do we send a fix-patch or send a new series? > + } > + > + if (storagebits > 16) > + *val = sign_extend32(st->data.buf32[ch], 17); > + else > + *val = sign_extend32(st->data.buf16[ch], 15); > > error_ret: > gpiod_set_value(st->gpio_convst, 0); > @@ -266,7 +474,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; > @@ -349,6 +557,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. > @@ -414,6 +634,20 @@ 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, > + .scale_setup_cb = ad7606c_16bit_chan_scale_setup, > + .oversampling_avail = ad7606_oversampling_avail, > + .oversampling_num = ARRAY_SIZE(ad7606_oversampling_avail), > + }, > + [ID_AD7606C_18] = { > + .channels = ad7606_channels_18bit, > + .num_channels = 9, > + .scale_setup_cb = ad7606c_18bit_chan_scale_setup, > + .oversampling_avail = ad7606_oversampling_avail, > + .oversampling_num = ARRAY_SIZE(ad7606_oversampling_avail), > + }, > [ID_AD7616] = { > .channels = ad7616_channels, > .num_channels = 17, > @@ -586,7 +820,7 @@ 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 int ad7606_sw_mode_setup(struct iio_dev *indio_dev, unsigned int id) > { > struct ad7606_state *st = iio_priv(indio_dev); > > @@ -604,13 +838,24 @@ static int ad7606_chan_scales_setup(struct iio_dev *indio_dev) > { > unsigned int num_channels = indio_dev->num_channels - 1; > struct ad7606_state *st = iio_priv(indio_dev); > + struct iio_chan_spec *chans; > + size_t size; > int ch, ret; > > + /* Clone IIO channels, since some may be differential */ > + size = indio_dev->num_channels * sizeof(*indio_dev->channels); > + chans = devm_kzalloc(st->dev, size, GFP_KERNEL); > + if (!chans) > + return -ENOMEM; > + > + memcpy(chans, indio_dev->channels, size); > + indio_dev->channels = chans; > + > for (ch = 0; ch < num_channels; ch++) { > struct ad7606_chan_scale *cs; > int i; > > - ret = st->chip_info->scale_setup_cb(st, ch); > + ret = st->chip_info->scale_setup_cb(st, &chans[ch + 1], ch); > if (ret) > return ret; > > @@ -698,7 +943,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 25e84efd15c3..14ee75aa225b 100644 > --- a/drivers/iio/adc/ad7606.h > +++ b/drivers/iio/adc/ad7606.h > @@ -63,7 +63,8 @@ > > struct ad7606_state; > > -typedef int (*ad7606_scale_setup_cb_t)(struct ad7606_state *st, int ch); > +typedef int (*ad7606_scale_setup_cb_t)(struct ad7606_state *st, > + struct iio_chan_spec *chan, int ch); > > /** > * struct ad7606_chip_info - chip specific information > @@ -94,6 +95,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_SCALES 16 > @@ -102,6 +105,7 @@ struct ad7606_chan_scale { > int scale_avail_show[AD760X_MAX_SCALE_SHOW]; > unsigned int num_scales; > unsigned int range; > + unsigned int reg_offset; > }; > > /** > @@ -158,9 +162,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 { > + u16 buf16[20]; > + u32 buf32[10]; > + } data __aligned(IIO_DMA_MINALIGN); > __be16 d16[2]; > }; > > @@ -201,6 +209,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..143440e73aab 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 * sizeof(u32), > + .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" }, > { } > }; > -- > 2.46.0 >
On 10/1/24 3:26 AM, Alexandru Ardelean wrote: > On Thu, Sep 19, 2024 at 4:05 PM Alexandru Ardelean > <aardelean@baylibre.com> wrote: >> ... >> @@ -153,7 +349,19 @@ static int ad7606_scan_direct(struct iio_dev *indio_dev, unsigned int ch, >> if (ret) >> goto error_ret; >> >> - *val = sign_extend32(st->data[ch], 15); >> + chan = &indio_dev->channels[ch + 1]; >> + if (chan->scan_type.sign == 'u') { >> + if (storagebits > 16) >> + *val = st->data.buf32[ch]; >> + else >> + *val = st->data.buf16[ch]; >> + return 0; > > Arrggh... > I messed up here. > Guillaume found a bug here, where this should be "goto error_ret" or > do an "if () {} else {}" > How should we do it here? > > Do we send a fix-patch or send a new series? > Since this patch is already applied, just follow up with another patch with a Fixes: tag.
On Tue, 1 Oct 2024 08:42:23 -0500 David Lechner <dlechner@baylibre.com> wrote: > On 10/1/24 3:26 AM, Alexandru Ardelean wrote: > > On Thu, Sep 19, 2024 at 4:05 PM Alexandru Ardelean > > <aardelean@baylibre.com> wrote: > >> > > ... > > >> @@ -153,7 +349,19 @@ static int ad7606_scan_direct(struct iio_dev *indio_dev, unsigned int ch, > >> if (ret) > >> goto error_ret; > >> > >> - *val = sign_extend32(st->data[ch], 15); > >> + chan = &indio_dev->channels[ch + 1]; > >> + if (chan->scan_type.sign == 'u') { > >> + if (storagebits > 16) > >> + *val = st->data.buf32[ch]; > >> + else > >> + *val = st->data.buf16[ch]; > >> + return 0; > > > > Arrggh... > > I messed up here. > > Guillaume found a bug here, where this should be "goto error_ret" or > > do an "if () {} else {}" > > How should we do it here? if / else. Goto an error label when it's not an error would be horrible! > > > > Do we send a fix-patch or send a new series? > > > > Since this patch is already applied, just follow up with another > patch with a Fixes: tag. I sometimes tweak these sort of things if I haven't pushed out as togreg yet (or they are really bad!) but in this case I'm not 100% sure what the comment is, so I'll just apply a fix on top. So David is entirely correct in general but by luck of timing this time I'll tweak it. Please check the result in iio.git/testing I'll hold off pushing that out as togreg until at least end of tomorrow. One more day o Jonathan > > >
On Tue, Oct 1, 2024 at 9:41 PM Jonathan Cameron <jic23@kernel.org> wrote: > > On Tue, 1 Oct 2024 08:42:23 -0500 > David Lechner <dlechner@baylibre.com> wrote: > > > On 10/1/24 3:26 AM, Alexandru Ardelean wrote: > > > On Thu, Sep 19, 2024 at 4:05 PM Alexandru Ardelean > > > <aardelean@baylibre.com> wrote: > > >> > > > > ... > > > > >> @@ -153,7 +349,19 @@ static int ad7606_scan_direct(struct iio_dev *indio_dev, unsigned int ch, > > >> if (ret) > > >> goto error_ret; > > >> > > >> - *val = sign_extend32(st->data[ch], 15); > > >> + chan = &indio_dev->channels[ch + 1]; > > >> + if (chan->scan_type.sign == 'u') { > > >> + if (storagebits > 16) > > >> + *val = st->data.buf32[ch]; > > >> + else > > >> + *val = st->data.buf16[ch]; > > >> + return 0; > > > > > > Arrggh... > > > I messed up here. > > > Guillaume found a bug here, where this should be "goto error_ret" or > > > do an "if () {} else {}" > > > How should we do it here? > if / else. Goto an error label when it's not an error would be horrible! > > > > > > Do we send a fix-patch or send a new series? > > > > > > > Since this patch is already applied, just follow up with another > > patch with a Fixes: tag. > > I sometimes tweak these sort of things if I haven't pushed out > as togreg yet (or they are really bad!) but in this case I'm not > 100% sure what the comment is, so I'll just apply a fix on top. > > So David is entirely correct in general but by luck of timing > this time I'll tweak it. > > Please check the result in iio.git/testing > I'll hold off pushing that out as togreg until at least end of > tomorrow. One more day o The "return 0" needs to be removed in the driver. if (chan->scan_type.sign == 'u') { if (storagebits > 16) *val = st->data.buf32[ch]; else *val = st->data.buf16[ch]; - return 0; } else { if (storagebits > 16) *val = sign_extend32(st->data.buf32[ch], 17); else *val = sign_extend32(st->data.buf16[ch], 15); } > > Jonathan > > > > > > > > >
On Wed, 2 Oct 2024 09:12:09 +0300 Alexandru Ardelean <aardelean@baylibre.com> wrote: > On Tue, Oct 1, 2024 at 9:41 PM Jonathan Cameron <jic23@kernel.org> wrote: > > > > On Tue, 1 Oct 2024 08:42:23 -0500 > > David Lechner <dlechner@baylibre.com> wrote: > > > > > On 10/1/24 3:26 AM, Alexandru Ardelean wrote: > > > > On Thu, Sep 19, 2024 at 4:05 PM Alexandru Ardelean > > > > <aardelean@baylibre.com> wrote: > > > >> > > > > > > ... > > > > > > >> @@ -153,7 +349,19 @@ static int ad7606_scan_direct(struct iio_dev *indio_dev, unsigned int ch, > > > >> if (ret) > > > >> goto error_ret; > > > >> > > > >> - *val = sign_extend32(st->data[ch], 15); > > > >> + chan = &indio_dev->channels[ch + 1]; > > > >> + if (chan->scan_type.sign == 'u') { > > > >> + if (storagebits > 16) > > > >> + *val = st->data.buf32[ch]; > > > >> + else > > > >> + *val = st->data.buf16[ch]; > > > >> + return 0; > > > > > > > > Arrggh... > > > > I messed up here. > > > > Guillaume found a bug here, where this should be "goto error_ret" or > > > > do an "if () {} else {}" > > > > How should we do it here? > > if / else. Goto an error label when it's not an error would be horrible! > > > > > > > > Do we send a fix-patch or send a new series? > > > > > > > > > > Since this patch is already applied, just follow up with another > > > patch with a Fixes: tag. > > > > I sometimes tweak these sort of things if I haven't pushed out > > as togreg yet (or they are really bad!) but in this case I'm not > > 100% sure what the comment is, so I'll just apply a fix on top. > > > > So David is entirely correct in general but by luck of timing > > this time I'll tweak it. > > > > Please check the result in iio.git/testing > > I'll hold off pushing that out as togreg until at least end of > > tomorrow. One more day o > > The "return 0" needs to be removed in the driver. > > if (chan->scan_type.sign == 'u') { > if (storagebits > 16) > *val = st->data.buf32[ch]; > else > *val = st->data.buf16[ch]; > - return 0; Doh!. Just goes to show why I shouldn't just edit these things. Stupid mistake. I'll fix when on right machine. Jonathan > } else { > if (storagebits > 16) > *val = sign_extend32(st->data.buf32[ch], 17); > else > *val = sign_extend32(st->data.buf16[ch], 15); > } > > > > > > > Jonathan > > > > > > > > > > > > > > > >
On Fri, 4 Oct 2024 14:54:30 +0100 Jonathan Cameron <Jonathan.Cameron@Huawei.com> wrote: > On Wed, 2 Oct 2024 09:12:09 +0300 > Alexandru Ardelean <aardelean@baylibre.com> wrote: > > > On Tue, Oct 1, 2024 at 9:41 PM Jonathan Cameron <jic23@kernel.org> wrote: > > > > > > On Tue, 1 Oct 2024 08:42:23 -0500 > > > David Lechner <dlechner@baylibre.com> wrote: > > > > > > > On 10/1/24 3:26 AM, Alexandru Ardelean wrote: > > > > > On Thu, Sep 19, 2024 at 4:05 PM Alexandru Ardelean > > > > > <aardelean@baylibre.com> wrote: > > > > >> > > > > > > > > ... > > > > > > > > >> @@ -153,7 +349,19 @@ static int ad7606_scan_direct(struct iio_dev *indio_dev, unsigned int ch, > > > > >> if (ret) > > > > >> goto error_ret; > > > > >> > > > > >> - *val = sign_extend32(st->data[ch], 15); > > > > >> + chan = &indio_dev->channels[ch + 1]; > > > > >> + if (chan->scan_type.sign == 'u') { > > > > >> + if (storagebits > 16) > > > > >> + *val = st->data.buf32[ch]; > > > > >> + else > > > > >> + *val = st->data.buf16[ch]; > > > > >> + return 0; > > > > > > > > > > Arrggh... > > > > > I messed up here. > > > > > Guillaume found a bug here, where this should be "goto error_ret" or > > > > > do an "if () {} else {}" > > > > > How should we do it here? > > > if / else. Goto an error label when it's not an error would be horrible! > > > > > > > > > > Do we send a fix-patch or send a new series? > > > > > > > > > > > > > Since this patch is already applied, just follow up with another > > > > patch with a Fixes: tag. > > > > > > I sometimes tweak these sort of things if I haven't pushed out > > > as togreg yet (or they are really bad!) but in this case I'm not > > > 100% sure what the comment is, so I'll just apply a fix on top. > > > > > > So David is entirely correct in general but by luck of timing > > > this time I'll tweak it. > > > > > > Please check the result in iio.git/testing > > > I'll hold off pushing that out as togreg until at least end of > > > tomorrow. One more day o > > > > The "return 0" needs to be removed in the driver. > > > > if (chan->scan_type.sign == 'u') { > > if (storagebits > 16) > > *val = st->data.buf32[ch]; > > else > > *val = st->data.buf16[ch]; > > - return 0; > Doh!. Just goes to show why I shouldn't just edit these things. > Stupid mistake. I'll fix when on right machine. hopefully now done J > > Jonathan > > > } else { > > if (storagebits > 16) > > *val = sign_extend32(st->data.buf32[ch], 17); > > else > > *val = sign_extend32(st->data.buf16[ch], 15); > > } > > > > > > > > > > > > Jonathan > > > > > > > > > > > > > > > > > > > > > > > >
On Sun, Oct 6, 2024 at 1:56 PM Jonathan Cameron <jic23@kernel.org> wrote: > > On Fri, 4 Oct 2024 14:54:30 +0100 > Jonathan Cameron <Jonathan.Cameron@Huawei.com> wrote: > > > On Wed, 2 Oct 2024 09:12:09 +0300 > > Alexandru Ardelean <aardelean@baylibre.com> wrote: > > > > > On Tue, Oct 1, 2024 at 9:41 PM Jonathan Cameron <jic23@kernel.org> wrote: > > > > > > > > On Tue, 1 Oct 2024 08:42:23 -0500 > > > > David Lechner <dlechner@baylibre.com> wrote: > > > > > > > > > On 10/1/24 3:26 AM, Alexandru Ardelean wrote: > > > > > > On Thu, Sep 19, 2024 at 4:05 PM Alexandru Ardelean > > > > > > <aardelean@baylibre.com> wrote: > > > > > >> > > > > > > > > > > ... > > > > > > > > > > >> @@ -153,7 +349,19 @@ static int ad7606_scan_direct(struct iio_dev *indio_dev, unsigned int ch, > > > > > >> if (ret) > > > > > >> goto error_ret; > > > > > >> > > > > > >> - *val = sign_extend32(st->data[ch], 15); > > > > > >> + chan = &indio_dev->channels[ch + 1]; > > > > > >> + if (chan->scan_type.sign == 'u') { > > > > > >> + if (storagebits > 16) > > > > > >> + *val = st->data.buf32[ch]; > > > > > >> + else > > > > > >> + *val = st->data.buf16[ch]; > > > > > >> + return 0; > > > > > > > > > > > > Arrggh... > > > > > > I messed up here. > > > > > > Guillaume found a bug here, where this should be "goto error_ret" or > > > > > > do an "if () {} else {}" > > > > > > How should we do it here? > > > > if / else. Goto an error label when it's not an error would be horrible! > > > > > > > > > > > > Do we send a fix-patch or send a new series? > > > > > > > > > > > > > > > > Since this patch is already applied, just follow up with another > > > > > patch with a Fixes: tag. > > > > > > > > I sometimes tweak these sort of things if I haven't pushed out > > > > as togreg yet (or they are really bad!) but in this case I'm not > > > > 100% sure what the comment is, so I'll just apply a fix on top. > > > > > > > > So David is entirely correct in general but by luck of timing > > > > this time I'll tweak it. > > > > > > > > Please check the result in iio.git/testing > > > > I'll hold off pushing that out as togreg until at least end of > > > > tomorrow. One more day o > > > > > > The "return 0" needs to be removed in the driver. > > > > > > if (chan->scan_type.sign == 'u') { > > > if (storagebits > 16) > > > *val = st->data.buf32[ch]; > > > else > > > *val = st->data.buf16[ch]; > > > - return 0; > > Doh!. Just goes to show why I shouldn't just edit these things. > > Stupid mistake. I'll fix when on right machine. > hopefully now done Looks good now. Apologies for the slow reply. Thanks Alex > > J > > > > Jonathan > > > > > } else { > > > if (storagebits > 16) > > > *val = sign_extend32(st->data.buf32[ch], 17); > > > else > > > *val = sign_extend32(st->data.buf16[ch], 15); > > > } > > > > > > > > > > > > > > > > > Jonathan > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > >
On 10/1/24 10:26, Alexandru Ardelean wrote: > On Thu, Sep 19, 2024 at 4:05 PM Alexandru Ardelean > <aardelean@baylibre.com> 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: https://www.analog.com/media/en/technical-documentation/data-sheets/ad7606c-16.pdf >> Datasheet: 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 | 263 +++++++++++++++++++++++++++++++++-- >> drivers/iio/adc/ad7606.h | 16 ++- >> drivers/iio/adc/ad7606_spi.c | 55 ++++++++ >> 3 files changed, 322 insertions(+), 12 deletions(-) >> >> diff --git a/drivers/iio/adc/ad7606.c b/drivers/iio/adc/ad7606.c >> index b909ee14fd81..f04e5660d2f8 100644 >> --- a/drivers/iio/adc/ad7606.c >> +++ b/drivers/iio/adc/ad7606.c >> @@ -36,6 +36,33 @@ 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_16bit_single_ended_unipolar_scale_avail[3] = { >> + 76294, 152588, 190735, >> +}; >> + >> +static const unsigned int ad7606c_16bit_single_ended_bipolar_scale_avail[5] = { >> + 76294, 152588, 190735, 305176, 381470 >> +}; >> + >> +static const unsigned int ad7606c_16bit_differential_bipolar_scale_avail[4] = { >> + 152588, 305176, 381470, 610352 >> +}; >> + >> +static const unsigned int ad7606c_18bit_single_ended_unipolar_scale_avail[3] = { >> + 19073, 38147, 47684 >> +}; >> + >> +static const unsigned int ad7606c_18bit_single_ended_bipolar_scale_avail[5] = { >> + 19073, 38147, 47684, 76294, 95367 >> +}; >> + >> +static const unsigned int ad7606c_18bit_differential_bipolar_scale_avail[4] = { >> + 38147, 76294, 95367, 152588 >> +}; >> >> static const unsigned int ad7606_16bit_sw_scale_avail[3] = { >> 76293, 152588, 305176 >> @@ -62,7 +89,8 @@ int ad7606_reset(struct ad7606_state *st) >> } >> EXPORT_SYMBOL_NS_GPL(ad7606_reset, IIO_AD7606); >> >> -static int ad7606_16bit_chan_scale_setup(struct ad7606_state *st, int ch) >> +static int ad7606_16bit_chan_scale_setup(struct ad7606_state *st, >> + struct iio_chan_spec *chan, int ch) >> { >> struct ad7606_chan_scale *cs = &st->chan_scales[ch]; >> >> @@ -83,6 +111,173 @@ static int ad7606_16bit_chan_scale_setup(struct ad7606_state *st, int ch) >> return 0; >> } >> >> +static int ad7606_get_chan_config(struct ad7606_state *st, int ch, >> + bool *bipolar, bool *differential) >> +{ >> + unsigned int num_channels = st->chip_info->num_channels - 1; >> + struct device *dev = st->dev; >> + int ret; >> + >> + *bipolar = false; >> + *differential = false; >> + >> + device_for_each_child_node_scoped(dev, child) { >> + u32 pins[2]; >> + int reg; >> + >> + ret = fwnode_property_read_u32(child, "reg", ®); >> + if (ret) >> + continue; >> + >> + /* channel number (here) is from 1 to num_channels */ >> + if (reg == 0 || reg > num_channels) { >> + dev_warn(dev, >> + "Invalid channel number (ignoring): %d\n", reg); >> + continue; >> + } >> + >> + if (reg != (ch + 1)) >> + continue; >> + >> + *bipolar = fwnode_property_read_bool(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] != reg || pins[1] != reg)) { >> + dev_err(dev, >> + "Differential pins must be the same as 'reg'"); >> + return -EINVAL; >> + } >> + >> + *differential = (ret == 0); >> + >> + if (*differential && !*bipolar) { >> + dev_err(dev, >> + "'bipolar' must be added for diff channel %d\n", >> + reg); >> + return -EINVAL; >> + } >> + >> + return 0; >> + } >> + >> + return 0; >> +} >> + >> +static int ad7606c_18bit_chan_scale_setup(struct ad7606_state *st, >> + struct iio_chan_spec *chan, int ch) >> +{ >> + struct ad7606_chan_scale *cs = &st->chan_scales[ch]; >> + bool bipolar, differential; >> + int ret; >> + >> + if (!st->sw_mode_en) { >> + cs->range = 0; >> + cs->scale_avail = ad7606_18bit_hw_scale_avail; >> + cs->num_scales = ARRAY_SIZE(ad7606_18bit_hw_scale_avail); >> + return 0; >> + } >> + >> + ret = ad7606_get_chan_config(st, ch, &bipolar, &differential); >> + if (ret) >> + return ret; >> + >> + if (differential) { >> + cs->scale_avail = ad7606c_18bit_differential_bipolar_scale_avail; >> + cs->num_scales = >> + ARRAY_SIZE(ad7606c_18bit_differential_bipolar_scale_avail); >> + /* Bipolar differential ranges start at 8 (b1000) */ >> + cs->reg_offset = 8; >> + cs->range = 1; >> + chan->differential = 1; >> + chan->channel2 = chan->channel; >> + >> + return 0; >> + } >> + >> + chan->differential = 0; >> + >> + if (bipolar) { >> + cs->scale_avail = ad7606c_18bit_single_ended_bipolar_scale_avail; >> + cs->num_scales = >> + ARRAY_SIZE(ad7606c_18bit_single_ended_bipolar_scale_avail); >> + /* Bipolar single-ended ranges start at 0 (b0000) */ >> + cs->reg_offset = 0; >> + cs->range = 3; >> + chan->scan_type.sign = 's'; >> + >> + return 0; >> + } >> + >> + cs->scale_avail = ad7606c_18bit_single_ended_unipolar_scale_avail; >> + cs->num_scales = >> + ARRAY_SIZE(ad7606c_18bit_single_ended_unipolar_scale_avail); >> + /* Unipolar single-ended ranges start at 5 (b0101) */ >> + cs->reg_offset = 5; >> + cs->range = 1; >> + chan->scan_type.sign = 'u'; >> + >> + return 0; >> +} >> + >> +static int ad7606c_16bit_chan_scale_setup(struct ad7606_state *st, >> + struct iio_chan_spec *chan, int ch) >> +{ >> + struct ad7606_chan_scale *cs = &st->chan_scales[ch]; >> + bool bipolar, differential; >> + int ret; >> + >> + if (!st->sw_mode_en) { >> + cs->range = 0; >> + cs->scale_avail = ad7606_16bit_hw_scale_avail; >> + cs->num_scales = ARRAY_SIZE(ad7606_16bit_hw_scale_avail); >> + return 0; >> + } >> + >> + ret = ad7606_get_chan_config(st, ch, &bipolar, &differential); >> + if (ret) >> + return ret; >> + >> + if (differential) { >> + cs->scale_avail = ad7606c_16bit_differential_bipolar_scale_avail; >> + cs->num_scales = >> + ARRAY_SIZE(ad7606c_16bit_differential_bipolar_scale_avail); >> + /* Bipolar differential ranges start at 8 (b1000) */ >> + cs->reg_offset = 8; >> + cs->range = 1; >> + chan->differential = 1; >> + chan->channel2 = chan->channel; >> + chan->scan_type.sign = 's'; >> + >> + return 0; >> + } >> + >> + chan->differential = 0; >> + >> + if (bipolar) { >> + cs->scale_avail = ad7606c_16bit_single_ended_bipolar_scale_avail; >> + cs->num_scales = >> + ARRAY_SIZE(ad7606c_16bit_single_ended_bipolar_scale_avail); >> + /* Bipolar single-ended ranges start at 0 (b0000) */ >> + cs->reg_offset = 0; >> + cs->range = 3; >> + chan->scan_type.sign = 's'; >> + >> + return 0; >> + } >> + >> + cs->scale_avail = ad7606c_16bit_single_ended_unipolar_scale_avail; >> + cs->num_scales = >> + ARRAY_SIZE(ad7606c_16bit_single_ended_unipolar_scale_avail); >> + /* Unipolar single-ended ranges start at 5 (b0101) */ >> + cs->reg_offset = 5; >> + cs->range = 1; >> + chan->scan_type.sign = 'u'; >> + >> + return 0; >> +} >> + >> static int ad7606_reg_access(struct iio_dev *indio_dev, >> unsigned int reg, >> unsigned int writeval, >> @@ -107,9 +302,8 @@ static int ad7606_reg_access(struct iio_dev *indio_dev, >> static int ad7606_read_samples(struct ad7606_state *st) >> { >> unsigned int num = st->chip_info->num_channels - 1; >> - u16 *data = st->data; >> >> - return st->bops->read_block(st->dev, num, data); >> + return st->bops->read_block(st->dev, num, &st->data); >> } >> >> static irqreturn_t ad7606_trigger_handler(int irq, void *p) >> @@ -125,7 +319,7 @@ static irqreturn_t ad7606_trigger_handler(int irq, void *p) >> 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, >> iio_get_time_ns(indio_dev)); >> error_ret: >> iio_trigger_notify_done(indio_dev->trig); >> @@ -139,6 +333,8 @@ 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; >> + const struct iio_chan_spec *chan; >> int ret; >> >> gpiod_set_value(st->gpio_convst, 1); >> @@ -153,7 +349,19 @@ static int ad7606_scan_direct(struct iio_dev *indio_dev, unsigned int ch, >> if (ret) >> goto error_ret; >> >> - *val = sign_extend32(st->data[ch], 15); >> + chan = &indio_dev->channels[ch + 1]; >> + if (chan->scan_type.sign == 'u') { >> + if (storagebits > 16) >> + *val = st->data.buf32[ch]; >> + else >> + *val = st->data.buf16[ch]; >> + return 0; > Arrggh... > I messed up here. > Guillaume found a bug here, where this should be "goto error_ret" or > do an "if () {} else {}" > How should we do it here? > > Do we send a fix-patch or send a new series? I can also send a fix in my next version of the "Add iio backend support" series which depends on this one. > > >> + } >> + >> + if (storagebits > 16) >> + *val = sign_extend32(st->data.buf32[ch], 17); >> + else >> + *val = sign_extend32(st->data.buf16[ch], 15); >> >> error_ret: >> gpiod_set_value(st->gpio_convst, 0); >> @@ -266,7 +474,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; >> @@ -349,6 +557,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. >> @@ -414,6 +634,20 @@ 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, >> + .scale_setup_cb = ad7606c_16bit_chan_scale_setup, >> + .oversampling_avail = ad7606_oversampling_avail, >> + .oversampling_num = ARRAY_SIZE(ad7606_oversampling_avail), >> + }, >> + [ID_AD7606C_18] = { >> + .channels = ad7606_channels_18bit, >> + .num_channels = 9, >> + .scale_setup_cb = ad7606c_18bit_chan_scale_setup, >> + .oversampling_avail = ad7606_oversampling_avail, >> + .oversampling_num = ARRAY_SIZE(ad7606_oversampling_avail), >> + }, >> [ID_AD7616] = { >> .channels = ad7616_channels, >> .num_channels = 17, >> @@ -586,7 +820,7 @@ 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 int ad7606_sw_mode_setup(struct iio_dev *indio_dev, unsigned int id) >> { >> struct ad7606_state *st = iio_priv(indio_dev); >> >> @@ -604,13 +838,24 @@ static int ad7606_chan_scales_setup(struct iio_dev *indio_dev) >> { >> unsigned int num_channels = indio_dev->num_channels - 1; >> struct ad7606_state *st = iio_priv(indio_dev); >> + struct iio_chan_spec *chans; >> + size_t size; >> int ch, ret; >> >> + /* Clone IIO channels, since some may be differential */ >> + size = indio_dev->num_channels * sizeof(*indio_dev->channels); >> + chans = devm_kzalloc(st->dev, size, GFP_KERNEL); >> + if (!chans) >> + return -ENOMEM; >> + >> + memcpy(chans, indio_dev->channels, size); >> + indio_dev->channels = chans; >> + >> for (ch = 0; ch < num_channels; ch++) { >> struct ad7606_chan_scale *cs; >> int i; >> >> - ret = st->chip_info->scale_setup_cb(st, ch); >> + ret = st->chip_info->scale_setup_cb(st, &chans[ch + 1], ch); >> if (ret) >> return ret; >> >> @@ -698,7 +943,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 25e84efd15c3..14ee75aa225b 100644 >> --- a/drivers/iio/adc/ad7606.h >> +++ b/drivers/iio/adc/ad7606.h >> @@ -63,7 +63,8 @@ >> >> struct ad7606_state; >> >> -typedef int (*ad7606_scale_setup_cb_t)(struct ad7606_state *st, int ch); >> +typedef int (*ad7606_scale_setup_cb_t)(struct ad7606_state *st, >> + struct iio_chan_spec *chan, int ch); >> >> /** >> * struct ad7606_chip_info - chip specific information >> @@ -94,6 +95,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_SCALES 16 >> @@ -102,6 +105,7 @@ struct ad7606_chan_scale { >> int scale_avail_show[AD760X_MAX_SCALE_SHOW]; >> unsigned int num_scales; >> unsigned int range; >> + unsigned int reg_offset; >> }; >> >> /** >> @@ -158,9 +162,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 { >> + u16 buf16[20]; >> + u32 buf32[10]; >> + } data __aligned(IIO_DMA_MINALIGN); >> __be16 d16[2]; >> }; >> >> @@ -201,6 +209,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..143440e73aab 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 * sizeof(u32), >> + .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" }, >> { } >> }; >> -- >> 2.46.0 >>
© 2016 - 2024 Red Hat, Inc.