Author | Tokens | Token Proportion | Commits | Commit Proportion |
---|---|---|---|---|
Antoniu Miclaus | 1697 | 100.00% | 1 | 100.00% |
Total | 1697 | 1 |
// 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");
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with Cregit http://github.com/cregit/cregit
Version 2.0-RC1