| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384 |
- // SPDX-License-Identifier: GPL-2.0-only
- /*
- * ADA4250 driver
- *
- * Copyright 2022 Analog Devices Inc.
- */
- #include <linux/bitfield.h>
- #include <linux/bits.h>
- #include <linux/device.h>
- #include <linux/iio/iio.h>
- #include <linux/module.h>
- #include <linux/regmap.h>
- #include <linux/regulator/consumer.h>
- #include <linux/spi/spi.h>
- #include <linux/types.h>
- #include <linux/units.h>
- /* ADA4250 Register Map */
- #define ADA4250_REG_GAIN_MUX 0x00
- #define ADA4250_REG_REFBUF_EN 0x01
- #define ADA4250_REG_RESET 0x02
- #define ADA4250_REG_SNSR_CAL_VAL 0x04
- #define ADA4250_REG_SNSR_CAL_CNFG 0x05
- #define ADA4250_REG_DIE_REV 0x18
- #define ADA4250_REG_CHIP_ID 0x19
- /* ADA4250_REG_GAIN_MUX Map */
- #define ADA4250_GAIN_MUX_MSK GENMASK(2, 0)
- /* ADA4250_REG_REFBUF Map */
- #define ADA4250_REFBUF_MSK BIT(0)
- /* ADA4250_REG_RESET Map */
- #define ADA4250_RESET_MSK BIT(0)
- /* ADA4250_REG_SNSR_CAL_VAL Map */
- #define ADA4250_CAL_CFG_BIAS_MSK GENMASK(7, 0)
- /* ADA4250_REG_SNSR_CAL_CNFG Bit Definition */
- #define ADA4250_BIAS_SET_MSK GENMASK(3, 2)
- #define ADA4250_RANGE_SET_MSK GENMASK(1, 0)
- /* Miscellaneous definitions */
- #define ADA4250_CHIP_ID 0x4250
- #define ADA4250_RANGE1 0
- #define ADA4250_RANGE4 3
- /* ADA4250 current bias set */
- enum ada4250_current_bias {
- ADA4250_BIAS_DISABLED,
- ADA4250_BIAS_BANDGAP,
- ADA4250_BIAS_AVDD,
- };
- struct ada4250_state {
- struct spi_device *spi;
- struct regmap *regmap;
- /* Protect against concurrent accesses to the device and data content */
- struct mutex lock;
- int avdd_uv;
- int offset_uv;
- u8 bias;
- u8 gain;
- bool refbuf_en;
- __le16 reg_val_16 __aligned(IIO_DMA_MINALIGN);
- };
- /* ADA4250 Current Bias Source Settings: Disabled, Bandgap Reference, AVDD */
- static const int calibbias_table[] = {0, 1, 2};
- /* ADA4250 Gain (V/V) values: 1, 2, 4, 8, 16, 32, 64, 128 */
- static const int hwgain_table[] = {1, 2, 4, 8, 16, 32, 64, 128};
- static const struct regmap_config ada4250_regmap_config = {
- .reg_bits = 8,
- .val_bits = 8,
- .read_flag_mask = BIT(7),
- .max_register = 0x1A,
- };
- static int ada4250_set_offset_uv(struct iio_dev *indio_dev,
- const struct iio_chan_spec *chan,
- int offset_uv)
- {
- struct ada4250_state *st = iio_priv(indio_dev);
- int i, ret, x[8], max_vos, min_vos, voltage_v, vlsb = 0;
- u8 offset_raw, range = ADA4250_RANGE1;
- u32 lsb_coeff[6] = {1333, 2301, 4283, 8289, 16311, 31599};
- if (st->bias == 0 || st->bias == 3)
- return -EINVAL;
- voltage_v = DIV_ROUND_CLOSEST(st->avdd_uv, MICRO);
- if (st->bias == ADA4250_BIAS_AVDD)
- x[0] = voltage_v;
- else
- x[0] = 5;
- x[1] = 126 * (x[0] - 1);
- for (i = 0; i < 6; i++)
- x[i + 2] = DIV_ROUND_CLOSEST(x[1] * 1000, lsb_coeff[i]);
- if (st->gain == 0)
- return -EINVAL;
- /*
- * Compute Range and Voltage per LSB for the Sensor Offset Calibration
- * Example of computation for Range 1 and Range 2 (Curren Bias Set = AVDD):
- * Range 1 Range 2
- * Gain | Max Vos(mV) | LSB(mV) | Max Vos(mV) | LSB(mV) |
- * 2 | X1*127 | X1=0.126(AVDD-1) | X1*3*127 | X1*3 |
- * 4 | X2*127 | X2=X1/1.3333 | X2*3*127 | X2*3 |
- * 8 | X3*127 | X3=X1/2.301 | X3*3*127 | X3*3 |
- * 16 | X4*127 | X4=X1/4.283 | X4*3*127 | X4*3 |
- * 32 | X5*127 | X5=X1/8.289 | X5*3*127 | X5*3 |
- * 64 | X6*127 | X6=X1/16.311 | X6*3*127 | X6*3 |
- * 128 | X7*127 | X7=X1/31.599 | X7*3*127 | X7*3 |
- */
- for (i = ADA4250_RANGE1; i <= ADA4250_RANGE4; i++) {
- max_vos = x[st->gain] * 127 * ((1 << (i + 1)) - 1);
- min_vos = -1 * max_vos;
- if (offset_uv > min_vos && offset_uv < max_vos) {
- range = i;
- vlsb = x[st->gain] * ((1 << (i + 1)) - 1);
- break;
- }
- }
- if (vlsb <= 0)
- return -EINVAL;
- offset_raw = DIV_ROUND_CLOSEST(abs(offset_uv), vlsb);
- mutex_lock(&st->lock);
- ret = regmap_update_bits(st->regmap, ADA4250_REG_SNSR_CAL_CNFG,
- ADA4250_RANGE_SET_MSK,
- FIELD_PREP(ADA4250_RANGE_SET_MSK, range));
- if (ret)
- goto exit;
- st->offset_uv = offset_raw * vlsb;
- /*
- * To set the offset calibration value, use bits [6:0] and bit 7 as the
- * polarity bit (set to "0" for a negative offset and "1" for a positive
- * offset).
- */
- if (offset_uv < 0) {
- offset_raw |= BIT(7);
- st->offset_uv *= (-1);
- }
- ret = regmap_write(st->regmap, ADA4250_REG_SNSR_CAL_VAL, offset_raw);
- exit:
- mutex_unlock(&st->lock);
- return ret;
- }
- static int ada4250_read_raw(struct iio_dev *indio_dev,
- struct iio_chan_spec const *chan,
- int *val, int *val2, long info)
- {
- struct ada4250_state *st = iio_priv(indio_dev);
- int ret;
- switch (info) {
- case IIO_CHAN_INFO_HARDWAREGAIN:
- ret = regmap_read(st->regmap, ADA4250_REG_GAIN_MUX, val);
- if (ret)
- return ret;
- *val = BIT(*val);
- return IIO_VAL_INT;
- case IIO_CHAN_INFO_OFFSET:
- *val = st->offset_uv;
- return IIO_VAL_INT;
- case IIO_CHAN_INFO_CALIBBIAS:
- ret = regmap_read(st->regmap, ADA4250_REG_SNSR_CAL_CNFG, val);
- if (ret)
- return ret;
- *val = FIELD_GET(ADA4250_BIAS_SET_MSK, *val);
- return IIO_VAL_INT;
- case IIO_CHAN_INFO_SCALE:
- *val = 1;
- *val2 = 1000000;
- return IIO_VAL_FRACTIONAL;
- default:
- return -EINVAL;
- }
- }
- static int ada4250_write_raw(struct iio_dev *indio_dev,
- struct iio_chan_spec const *chan,
- int val, int val2, long info)
- {
- struct ada4250_state *st = iio_priv(indio_dev);
- int ret;
- switch (info) {
- case IIO_CHAN_INFO_HARDWAREGAIN:
- ret = regmap_write(st->regmap, ADA4250_REG_GAIN_MUX,
- FIELD_PREP(ADA4250_GAIN_MUX_MSK, ilog2(val)));
- if (ret)
- return ret;
- st->gain = ilog2(val);
- return ret;
- case IIO_CHAN_INFO_OFFSET:
- return ada4250_set_offset_uv(indio_dev, chan, val);
- case IIO_CHAN_INFO_CALIBBIAS:
- ret = regmap_update_bits(st->regmap, ADA4250_REG_SNSR_CAL_CNFG,
- ADA4250_BIAS_SET_MSK,
- FIELD_PREP(ADA4250_BIAS_SET_MSK, val));
- if (ret)
- return ret;
- st->bias = val;
- return ret;
- default:
- return -EINVAL;
- }
- }
- static int ada4250_read_avail(struct iio_dev *indio_dev,
- struct iio_chan_spec const *chan,
- const int **vals, int *type, int *length,
- long mask)
- {
- switch (mask) {
- case IIO_CHAN_INFO_CALIBBIAS:
- *vals = calibbias_table;
- *type = IIO_VAL_INT;
- *length = ARRAY_SIZE(calibbias_table);
- return IIO_AVAIL_LIST;
- case IIO_CHAN_INFO_HARDWAREGAIN:
- *vals = hwgain_table;
- *type = IIO_VAL_INT;
- *length = ARRAY_SIZE(hwgain_table);
- return IIO_AVAIL_LIST;
- default:
- return -EINVAL;
- }
- }
- static int ada4250_reg_access(struct iio_dev *indio_dev,
- unsigned int reg,
- unsigned int write_val,
- unsigned int *read_val)
- {
- struct ada4250_state *st = iio_priv(indio_dev);
- if (read_val)
- return regmap_read(st->regmap, reg, read_val);
- else
- return regmap_write(st->regmap, reg, write_val);
- }
- static const struct iio_info ada4250_info = {
- .read_raw = ada4250_read_raw,
- .write_raw = ada4250_write_raw,
- .read_avail = &ada4250_read_avail,
- .debugfs_reg_access = &ada4250_reg_access,
- };
- static const struct iio_chan_spec ada4250_channels[] = {
- {
- .type = IIO_VOLTAGE,
- .output = 1,
- .indexed = 1,
- .channel = 0,
- .info_mask_separate = BIT(IIO_CHAN_INFO_HARDWAREGAIN) |
- BIT(IIO_CHAN_INFO_OFFSET) |
- BIT(IIO_CHAN_INFO_CALIBBIAS) |
- BIT(IIO_CHAN_INFO_SCALE),
- .info_mask_separate_available = BIT(IIO_CHAN_INFO_CALIBBIAS) |
- BIT(IIO_CHAN_INFO_HARDWAREGAIN),
- }
- };
- static int ada4250_init(struct ada4250_state *st)
- {
- struct device *dev = &st->spi->dev;
- int ret;
- u16 chip_id;
- st->refbuf_en = device_property_read_bool(dev, "adi,refbuf-enable");
- st->avdd_uv = devm_regulator_get_enable_read_voltage(dev, "avdd");
- if (st->avdd_uv < 0)
- return dev_err_probe(dev, st->avdd_uv,
- "failed to get the AVDD voltage\n");
- ret = regmap_write(st->regmap, ADA4250_REG_RESET,
- FIELD_PREP(ADA4250_RESET_MSK, 1));
- if (ret)
- return ret;
- ret = regmap_bulk_read(st->regmap, ADA4250_REG_CHIP_ID, &st->reg_val_16,
- sizeof(st->reg_val_16));
- if (ret)
- return ret;
- chip_id = le16_to_cpu(st->reg_val_16);
- if (chip_id != ADA4250_CHIP_ID)
- dev_info(dev, "Invalid chip ID: 0x%02X.\n", chip_id);
- return regmap_write(st->regmap, ADA4250_REG_REFBUF_EN,
- FIELD_PREP(ADA4250_REFBUF_MSK, st->refbuf_en));
- }
- static int ada4250_probe(struct spi_device *spi)
- {
- struct iio_dev *indio_dev;
- struct regmap *regmap;
- struct ada4250_state *st;
- int ret;
- indio_dev = devm_iio_device_alloc(&spi->dev, sizeof(*st));
- if (!indio_dev)
- return -ENOMEM;
- regmap = devm_regmap_init_spi(spi, &ada4250_regmap_config);
- if (IS_ERR(regmap))
- return PTR_ERR(regmap);
- st = iio_priv(indio_dev);
- st->regmap = regmap;
- st->spi = spi;
- indio_dev->info = &ada4250_info;
- indio_dev->name = "ada4250";
- indio_dev->channels = ada4250_channels;
- indio_dev->num_channels = ARRAY_SIZE(ada4250_channels);
- mutex_init(&st->lock);
- ret = ada4250_init(st);
- if (ret)
- return dev_err_probe(&spi->dev, ret, "ADA4250 init failed\n");
- return devm_iio_device_register(&spi->dev, indio_dev);
- }
- static const struct spi_device_id ada4250_id[] = {
- { "ada4250", 0 },
- { }
- };
- MODULE_DEVICE_TABLE(spi, ada4250_id);
- static const struct of_device_id ada4250_of_match[] = {
- { .compatible = "adi,ada4250" },
- { }
- };
- MODULE_DEVICE_TABLE(of, ada4250_of_match);
- static struct spi_driver ada4250_driver = {
- .driver = {
- .name = "ada4250",
- .of_match_table = ada4250_of_match,
- },
- .probe = ada4250_probe,
- .id_table = ada4250_id,
- };
- module_spi_driver(ada4250_driver);
- MODULE_AUTHOR("Antoniu Miclaus <antoniu.miclaus@analog.com");
- MODULE_DESCRIPTION("Analog Devices ADA4250");
- MODULE_LICENSE("GPL v2");
|