|  | // 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 <asm/unaligned.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; | 
|  | struct regulator	*reg; | 
|  | /* Protect against concurrent accesses to the device and data content */ | 
|  | struct mutex		lock; | 
|  | u8			bias; | 
|  | u8			gain; | 
|  | int			offset_uv; | 
|  | bool			refbuf_en; | 
|  | }; | 
|  |  | 
|  | /* 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 = regulator_get_voltage(st->reg); | 
|  | voltage_v = DIV_ROUND_CLOSEST(voltage_v, 1000000); | 
|  |  | 
|  | 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 void ada4250_reg_disable(void *data) | 
|  | { | 
|  | regulator_disable(data); | 
|  | } | 
|  |  | 
|  | static int ada4250_init(struct ada4250_state *st) | 
|  | { | 
|  | int ret; | 
|  | u16 chip_id; | 
|  | u8 data[2] __aligned(8) = {}; | 
|  | struct spi_device *spi = st->spi; | 
|  |  | 
|  | st->refbuf_en = device_property_read_bool(&spi->dev, "adi,refbuf-enable"); | 
|  |  | 
|  | st->reg = devm_regulator_get(&spi->dev, "avdd"); | 
|  | if (IS_ERR(st->reg)) | 
|  | return dev_err_probe(&spi->dev, PTR_ERR(st->reg), | 
|  | "failed to get the AVDD voltage\n"); | 
|  |  | 
|  | ret = regulator_enable(st->reg); | 
|  | if (ret) { | 
|  | dev_err(&spi->dev, "Failed to enable specified AVDD supply\n"); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | ret = devm_add_action_or_reset(&spi->dev, ada4250_reg_disable, st->reg); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | 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, data, 2); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | chip_id = get_unaligned_le16(data); | 
|  |  | 
|  | if (chip_id != ADA4250_CHIP_ID) { | 
|  | dev_err(&spi->dev, "Invalid chip ID.\n"); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | 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) { | 
|  | dev_err(&spi->dev, "ADA4250 init failed\n"); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | 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"); |