> On Mon, 1 Jan 2018 19:54:44 +0100 > Lorenzo Bianconi <lorenzo.bianconi@xxxxxxxxxx> wrote: > >> Introduce regmap API support to access to i2c/spi bus instead of >> using a custom support. Set max bulk read to >> (32 / SAMPLE_SIZE) * SAMPLE_SIZE since spi_write_then_read() used in >> regmap_spi indicates that is the max buffer length to use in order to >> avoid a kmalloc for each bus access. >> Remove lock mutex since concurrency is already managed by regmap API >> >> Signed-off-by: Lorenzo Bianconi <lorenzo.bianconi@xxxxxxxxxx> > > There are a couple of sparse warnings unconnected to this patch that I'd > love it if you would clean up: > > CHECK drivers/iio/imu/st_lsm6dsx/st_lsm6dsx_buffer.c > drivers/iio/imu/st_lsm6dsx/st_lsm6dsx_buffer.c:250:17: warning: Variable length array is used. > drivers/iio/imu/st_lsm6dsx/st_lsm6dsx_buffer.c:283:55: error: cannot size expression > > I would fix that one by just making the array the maximum size it ever needs to > be. hw->sip would be at most 33 (acc_odr = 416 and gyro_odr = 13), so pattern_len = 198 (and it will be bigger since I would like to add hw timestamping) so I am wondering if it is better to pre-allocate it on the heap (maybe in st_lsm6dsx_hw data structure) instead of placing it on the stack, something like: struct st_lsm6dsx_hw { u8 data[MAX_SIZE]; } what do you think? > > On the FIELD_PREP not working unless mask is available at compile time, > I wonder if a bit of macro magic would allow us to fall back transparently > to a version that works at runtime if the compiler can't figure out the > mask. > Any hint? :) Regards, Lorenzo > Would make it more generally useful but would need a lot of care. > > Otherwise looks good to me. > > Thanks, > > Jonathan > >> --- >> drivers/iio/imu/st_lsm6dsx/Kconfig | 2 + >> drivers/iio/imu/st_lsm6dsx/st_lsm6dsx.h | 31 ++------ >> drivers/iio/imu/st_lsm6dsx/st_lsm6dsx_buffer.c | 81 ++++++++++++------- >> drivers/iio/imu/st_lsm6dsx/st_lsm6dsx_core.c | 103 ++++++++++--------------- >> drivers/iio/imu/st_lsm6dsx/st_lsm6dsx_i2c.c | 55 ++++--------- >> drivers/iio/imu/st_lsm6dsx/st_lsm6dsx_spi.c | 70 ++++------------- >> 6 files changed, 131 insertions(+), 211 deletions(-) >> >> diff --git a/drivers/iio/imu/st_lsm6dsx/Kconfig b/drivers/iio/imu/st_lsm6dsx/Kconfig >> index e57337159b57..14f2eb6e9fb7 100644 >> --- a/drivers/iio/imu/st_lsm6dsx/Kconfig >> +++ b/drivers/iio/imu/st_lsm6dsx/Kconfig >> @@ -16,7 +16,9 @@ config IIO_ST_LSM6DSX >> config IIO_ST_LSM6DSX_I2C >> tristate >> depends on IIO_ST_LSM6DSX >> + select REGMAP_I2C >> >> config IIO_ST_LSM6DSX_SPI >> tristate >> depends on IIO_ST_LSM6DSX >> + select REGMAP_SPI >> diff --git a/drivers/iio/imu/st_lsm6dsx/st_lsm6dsx.h b/drivers/iio/imu/st_lsm6dsx/st_lsm6dsx.h >> index cebc6bd31b79..ccbe44cef41a 100644 >> --- a/drivers/iio/imu/st_lsm6dsx/st_lsm6dsx.h >> +++ b/drivers/iio/imu/st_lsm6dsx/st_lsm6dsx.h >> @@ -29,21 +29,9 @@ enum st_lsm6dsx_hw_id { >> >> #define ST_LSM6DSX_CHAN_SIZE 2 >> #define ST_LSM6DSX_SAMPLE_SIZE 6 >> - >> -#if defined(CONFIG_SPI_MASTER) >> -#define ST_LSM6DSX_RX_MAX_LENGTH 256 >> -#define ST_LSM6DSX_TX_MAX_LENGTH 8 >> - >> -struct st_lsm6dsx_transfer_buffer { >> - u8 rx_buf[ST_LSM6DSX_RX_MAX_LENGTH]; >> - u8 tx_buf[ST_LSM6DSX_TX_MAX_LENGTH] ____cacheline_aligned; >> -}; >> -#endif /* CONFIG_SPI_MASTER */ >> - >> -struct st_lsm6dsx_transfer_function { >> - int (*read)(struct device *dev, u8 addr, int len, u8 *data); >> - int (*write)(struct device *dev, u8 addr, int len, u8 *data); >> -}; >> +#define ST_LSM6DSX_MAX_WORD_LEN ((32 / ST_LSM6DSX_SAMPLE_SIZE) * \ >> + ST_LSM6DSX_SAMPLE_SIZE) >> +#define ST_LSM6DSX_SHIFT_VAL(val, mask) (((val) << __ffs(mask)) & (mask)) >> >> struct st_lsm6dsx_reg { >> u8 addr; >> @@ -127,8 +115,8 @@ struct st_lsm6dsx_sensor { >> /** >> * struct st_lsm6dsx_hw - ST IMU MEMS hw instance >> * @dev: Pointer to instance of struct device (I2C or SPI). >> + * @regmap: Register map of the device. >> * @irq: Device interrupt line (I2C or SPI). >> - * @lock: Mutex to protect read and write operations. >> * @fifo_lock: Mutex to prevent concurrent access to the hw FIFO. >> * @conf_lock: Mutex to prevent concurrent FIFO configuration update. >> * @fifo_mode: FIFO operating mode supported by the device. >> @@ -136,14 +124,12 @@ struct st_lsm6dsx_sensor { >> * @sip: Total number of samples (acc/gyro) in a given pattern. >> * @iio_devs: Pointers to acc/gyro iio_dev instances. >> * @settings: Pointer to the specific sensor settings in use. >> - * @tf: Transfer function structure used by I/O operations. >> - * @tb: Transfer buffers used by SPI I/O operations. >> */ >> struct st_lsm6dsx_hw { >> struct device *dev; >> + struct regmap *regmap; >> int irq; >> >> - struct mutex lock; >> struct mutex fifo_lock; >> struct mutex conf_lock; >> >> @@ -154,17 +140,12 @@ struct st_lsm6dsx_hw { >> struct iio_dev *iio_devs[ST_LSM6DSX_ID_MAX]; >> >> const struct st_lsm6dsx_settings *settings; >> - >> - const struct st_lsm6dsx_transfer_function *tf; >> -#if defined(CONFIG_SPI_MASTER) >> - struct st_lsm6dsx_transfer_buffer tb; >> -#endif /* CONFIG_SPI_MASTER */ >> }; >> >> extern const struct dev_pm_ops st_lsm6dsx_pm_ops; >> >> int st_lsm6dsx_probe(struct device *dev, int irq, int hw_id, const char *name, >> - const struct st_lsm6dsx_transfer_function *tf_ops); >> + struct regmap *regmap); >> int st_lsm6dsx_sensor_enable(struct st_lsm6dsx_sensor *sensor); >> int st_lsm6dsx_sensor_disable(struct st_lsm6dsx_sensor *sensor); >> int st_lsm6dsx_fifo_setup(struct st_lsm6dsx_hw *hw); >> diff --git a/drivers/iio/imu/st_lsm6dsx/st_lsm6dsx_buffer.c b/drivers/iio/imu/st_lsm6dsx/st_lsm6dsx_buffer.c >> index c899d658f6be..093f9750974a 100644 >> --- a/drivers/iio/imu/st_lsm6dsx/st_lsm6dsx_buffer.c >> +++ b/drivers/iio/imu/st_lsm6dsx/st_lsm6dsx_buffer.c >> @@ -30,6 +30,8 @@ >> #include <linux/iio/kfifo_buf.h> >> #include <linux/iio/iio.h> >> #include <linux/iio/buffer.h> >> +#include <linux/regmap.h> >> +#include <linux/bitfield.h> >> >> #include <linux/platform_data/st_sensors_pdata.h> >> >> @@ -120,8 +122,10 @@ static int st_lsm6dsx_update_decimators(struct st_lsm6dsx_hw *hw) >> >> dec_reg = &hw->settings->decimator[sensor->id]; >> if (dec_reg->addr) { >> - err = st_lsm6dsx_write_with_mask(hw, dec_reg->addr, >> - dec_reg->mask, data); >> + int val = ST_LSM6DSX_SHIFT_VAL(data, dec_reg->mask); >> + >> + err = regmap_update_bits(hw->regmap, dec_reg->addr, >> + dec_reg->mask, val); >> if (err < 0) >> return err; >> } >> @@ -137,8 +141,10 @@ int st_lsm6dsx_set_fifo_mode(struct st_lsm6dsx_hw *hw, >> { >> int err; >> >> - err = st_lsm6dsx_write_with_mask(hw, ST_LSM6DSX_REG_FIFO_MODE_ADDR, >> - ST_LSM6DSX_FIFO_MODE_MASK, fifo_mode); >> + err = regmap_update_bits(hw->regmap, ST_LSM6DSX_REG_FIFO_MODE_ADDR, >> + ST_LSM6DSX_FIFO_MODE_MASK, >> + FIELD_PREP(ST_LSM6DSX_FIFO_MODE_MASK, >> + fifo_mode)); >> if (err < 0) >> return err; >> >> @@ -154,8 +160,9 @@ static int st_lsm6dsx_set_fifo_odr(struct st_lsm6dsx_sensor *sensor, >> u8 data; >> >> data = hw->enable_mask ? ST_LSM6DSX_MAX_FIFO_ODR_VAL : 0; >> - return st_lsm6dsx_write_with_mask(hw, ST_LSM6DSX_REG_FIFO_MODE_ADDR, >> - ST_LSM6DSX_FIFO_ODR_MASK, data); >> + return regmap_update_bits(hw->regmap, ST_LSM6DSX_REG_FIFO_MODE_ADDR, >> + ST_LSM6DSX_FIFO_ODR_MASK, >> + FIELD_PREP(ST_LSM6DSX_FIFO_ODR_MASK, data)); >> } >> >> int st_lsm6dsx_update_watermark(struct st_lsm6dsx_sensor *sensor, u16 watermark) >> @@ -163,9 +170,8 @@ int st_lsm6dsx_update_watermark(struct st_lsm6dsx_sensor *sensor, u16 watermark) >> u16 fifo_watermark = ~0, cur_watermark, sip = 0, fifo_th_mask; >> struct st_lsm6dsx_hw *hw = sensor->hw; >> struct st_lsm6dsx_sensor *cur_sensor; >> + int i, err, data; >> __le16 wdata; >> - int i, err; >> - u8 data; >> >> for (i = 0; i < ST_LSM6DSX_ID_MAX; i++) { >> cur_sensor = iio_priv(hw->iio_devs[i]); >> @@ -187,24 +193,42 @@ int st_lsm6dsx_update_watermark(struct st_lsm6dsx_sensor *sensor, u16 watermark) >> fifo_watermark = (fifo_watermark / sip) * sip; >> fifo_watermark = fifo_watermark * hw->settings->fifo_ops.th_wl; >> >> - mutex_lock(&hw->lock); >> - >> - err = hw->tf->read(hw->dev, hw->settings->fifo_ops.fifo_th.addr + 1, >> - sizeof(data), &data); >> + err = regmap_read(hw->regmap, hw->settings->fifo_ops.fifo_th.addr + 1, >> + &data); >> if (err < 0) >> - goto out; >> + return err; >> >> fifo_th_mask = hw->settings->fifo_ops.fifo_th.mask; >> fifo_watermark = ((data << 8) & ~fifo_th_mask) | >> (fifo_watermark & fifo_th_mask); >> >> wdata = cpu_to_le16(fifo_watermark); >> - err = hw->tf->write(hw->dev, hw->settings->fifo_ops.fifo_th.addr, >> - sizeof(wdata), (u8 *)&wdata); >> -out: >> - mutex_unlock(&hw->lock); >> + return regmap_bulk_write(hw->regmap, >> + hw->settings->fifo_ops.fifo_th.addr, >> + &wdata, sizeof(wdata)); >> +} >> >> - return err < 0 ? err : 0; >> +/* >> + * Set max bulk read to ST_LSM6DSX_MAX_WORD_LEN in order to avoid >> + * a kmalloc for each bus access >> + */ >> +static inline int st_lsm6dsx_read_block(struct st_lsm6dsx_hw *hw, u8 *data, >> + unsigned int data_len) >> +{ >> + unsigned int word_len, read_len = 0; >> + int err; >> + >> + while (read_len < data_len) { >> + word_len = min_t(unsigned int, data_len - read_len, >> + ST_LSM6DSX_MAX_WORD_LEN); >> + err = regmap_bulk_read(hw->regmap, >> + ST_LSM6DSX_REG_FIFO_OUTL_ADDR, >> + data + read_len, word_len); >> + if (err < 0) >> + return err; >> + read_len += word_len; >> + } >> + return 0; >> } >> >> /** >> @@ -226,8 +250,9 @@ static int st_lsm6dsx_read_fifo(struct st_lsm6dsx_hw *hw) >> u8 buff[pattern_len]; >> __le16 fifo_status; >> >> - err = hw->tf->read(hw->dev, hw->settings->fifo_ops.fifo_diff.addr, >> - sizeof(fifo_status), (u8 *)&fifo_status); >> + err = regmap_bulk_read(hw->regmap, >> + hw->settings->fifo_ops.fifo_diff.addr, >> + &fifo_status, sizeof(fifo_status)); >> if (err < 0) >> return err; >> >> @@ -255,8 +280,7 @@ static int st_lsm6dsx_read_fifo(struct st_lsm6dsx_hw *hw) >> samples); >> >> for (read_len = 0; read_len < fifo_len; read_len += pattern_len) { >> - err = hw->tf->read(hw->dev, ST_LSM6DSX_REG_FIFO_OUTL_ADDR, >> - sizeof(buff), buff); >> + err = st_lsm6dsx_read_block(hw, buff, sizeof(buff)); >> if (err < 0) >> return err; >> >> @@ -449,17 +473,20 @@ int st_lsm6dsx_fifo_setup(struct st_lsm6dsx_hw *hw) >> return -EINVAL; >> } >> >> - err = st_lsm6dsx_write_with_mask(hw, ST_LSM6DSX_REG_HLACTIVE_ADDR, >> - ST_LSM6DSX_REG_HLACTIVE_MASK, >> - irq_active_low); >> + err = regmap_update_bits(hw->regmap, ST_LSM6DSX_REG_HLACTIVE_ADDR, >> + ST_LSM6DSX_REG_HLACTIVE_MASK, >> + FIELD_PREP(ST_LSM6DSX_REG_HLACTIVE_MASK, >> + irq_active_low)); >> if (err < 0) >> return err; >> >> pdata = (struct st_sensors_platform_data *)hw->dev->platform_data; >> if ((np && of_property_read_bool(np, "drive-open-drain")) || >> (pdata && pdata->open_drain)) { >> - err = st_lsm6dsx_write_with_mask(hw, ST_LSM6DSX_REG_PP_OD_ADDR, >> - ST_LSM6DSX_REG_PP_OD_MASK, 1); >> + err = regmap_update_bits(hw->regmap, ST_LSM6DSX_REG_PP_OD_ADDR, >> + ST_LSM6DSX_REG_PP_OD_MASK, >> + FIELD_PREP(ST_LSM6DSX_REG_PP_OD_MASK, >> + 1)); >> if (err < 0) >> return err; >> >> diff --git a/drivers/iio/imu/st_lsm6dsx/st_lsm6dsx_core.c b/drivers/iio/imu/st_lsm6dsx/st_lsm6dsx_core.c >> index 4d43c956d676..819a85bb86ec 100644 >> --- a/drivers/iio/imu/st_lsm6dsx/st_lsm6dsx_core.c >> +++ b/drivers/iio/imu/st_lsm6dsx/st_lsm6dsx_core.c >> @@ -37,6 +37,8 @@ >> #include <linux/iio/iio.h> >> #include <linux/iio/sysfs.h> >> #include <linux/pm.h> >> +#include <linux/regmap.h> >> +#include <linux/bitfield.h> >> >> #include <linux/platform_data/st_sensors_pdata.h> >> >> @@ -277,36 +279,9 @@ static const struct iio_chan_spec st_lsm6dsx_gyro_channels[] = { >> IIO_CHAN_SOFT_TIMESTAMP(3), >> }; >> >> -int st_lsm6dsx_write_with_mask(struct st_lsm6dsx_hw *hw, u8 addr, u8 mask, >> - u8 val) >> -{ >> - u8 data; >> - int err; >> - >> - mutex_lock(&hw->lock); >> - >> - err = hw->tf->read(hw->dev, addr, sizeof(data), &data); >> - if (err < 0) { >> - dev_err(hw->dev, "failed to read %02x register\n", addr); >> - goto out; >> - } >> - >> - data = (data & ~mask) | ((val << __ffs(mask)) & mask); >> - >> - err = hw->tf->write(hw->dev, addr, sizeof(data), &data); >> - if (err < 0) >> - dev_err(hw->dev, "failed to write %02x register\n", addr); >> - >> -out: >> - mutex_unlock(&hw->lock); >> - >> - return err; >> -} >> - >> static int st_lsm6dsx_check_whoami(struct st_lsm6dsx_hw *hw, int id) >> { >> - int err, i, j; >> - u8 data; >> + int err, i, j, data; >> >> for (i = 0; i < ARRAY_SIZE(st_lsm6dsx_sensor_settings); i++) { >> for (j = 0; j < ST_LSM6DSX_MAX_ID; j++) { >> @@ -322,8 +297,7 @@ static int st_lsm6dsx_check_whoami(struct st_lsm6dsx_hw *hw, int id) >> return -ENODEV; >> } >> >> - err = hw->tf->read(hw->dev, ST_LSM6DSX_REG_WHOAMI_ADDR, sizeof(data), >> - &data); >> + err = regmap_read(hw->regmap, ST_LSM6DSX_REG_WHOAMI_ADDR, &data); >> if (err < 0) { >> dev_err(hw->dev, "failed to read whoami register\n"); >> return err; >> @@ -342,22 +316,22 @@ static int st_lsm6dsx_check_whoami(struct st_lsm6dsx_hw *hw, int id) >> static int st_lsm6dsx_set_full_scale(struct st_lsm6dsx_sensor *sensor, >> u32 gain) >> { >> - enum st_lsm6dsx_sensor_id id = sensor->id; >> + struct st_lsm6dsx_hw *hw = sensor->hw; >> + const struct st_lsm6dsx_reg *reg; >> int i, err; >> u8 val; >> >> for (i = 0; i < ST_LSM6DSX_FS_LIST_SIZE; i++) >> - if (st_lsm6dsx_fs_table[id].fs_avl[i].gain == gain) >> + if (st_lsm6dsx_fs_table[sensor->id].fs_avl[i].gain == gain) >> break; >> >> if (i == ST_LSM6DSX_FS_LIST_SIZE) >> return -EINVAL; >> >> - val = st_lsm6dsx_fs_table[id].fs_avl[i].val; >> - err = st_lsm6dsx_write_with_mask(sensor->hw, >> - st_lsm6dsx_fs_table[id].reg.addr, >> - st_lsm6dsx_fs_table[id].reg.mask, >> - val); >> + val = st_lsm6dsx_fs_table[sensor->id].fs_avl[i].val; >> + reg = &st_lsm6dsx_fs_table[sensor->id].reg; >> + err = regmap_update_bits(hw->regmap, reg->addr, reg->mask, >> + ST_LSM6DSX_SHIFT_VAL(val, reg->mask)); >> if (err < 0) >> return err; >> >> @@ -385,7 +359,8 @@ static int st_lsm6dsx_check_odr(struct st_lsm6dsx_sensor *sensor, u16 odr, >> >> static int st_lsm6dsx_set_odr(struct st_lsm6dsx_sensor *sensor, u16 odr) >> { >> - enum st_lsm6dsx_sensor_id id = sensor->id; >> + struct st_lsm6dsx_hw *hw = sensor->hw; >> + const struct st_lsm6dsx_reg *reg; >> int err; >> u8 val; >> >> @@ -393,10 +368,9 @@ static int st_lsm6dsx_set_odr(struct st_lsm6dsx_sensor *sensor, u16 odr) >> if (err < 0) >> return err; >> >> - return st_lsm6dsx_write_with_mask(sensor->hw, >> - st_lsm6dsx_odr_table[id].reg.addr, >> - st_lsm6dsx_odr_table[id].reg.mask, >> - val); >> + reg = &st_lsm6dsx_odr_table[sensor->id].reg; >> + return regmap_update_bits(hw->regmap, reg->addr, reg->mask, >> + ST_LSM6DSX_SHIFT_VAL(val, reg->mask)); >> } >> >> int st_lsm6dsx_sensor_enable(struct st_lsm6dsx_sensor *sensor) >> @@ -414,16 +388,17 @@ int st_lsm6dsx_sensor_enable(struct st_lsm6dsx_sensor *sensor) >> >> int st_lsm6dsx_sensor_disable(struct st_lsm6dsx_sensor *sensor) >> { >> - enum st_lsm6dsx_sensor_id id = sensor->id; >> + struct st_lsm6dsx_hw *hw = sensor->hw; >> + const struct st_lsm6dsx_reg *reg; >> int err; >> >> - err = st_lsm6dsx_write_with_mask(sensor->hw, >> - st_lsm6dsx_odr_table[id].reg.addr, >> - st_lsm6dsx_odr_table[id].reg.mask, 0); >> + reg = &st_lsm6dsx_odr_table[sensor->id].reg; >> + err = regmap_update_bits(hw->regmap, reg->addr, reg->mask, >> + ST_LSM6DSX_SHIFT_VAL(0, reg->mask)); >> if (err < 0) >> return err; >> >> - sensor->hw->enable_mask &= ~BIT(id); >> + sensor->hw->enable_mask &= ~BIT(sensor->id); >> >> return 0; >> } >> @@ -431,6 +406,7 @@ int st_lsm6dsx_sensor_disable(struct st_lsm6dsx_sensor *sensor) >> static int st_lsm6dsx_read_oneshot(struct st_lsm6dsx_sensor *sensor, >> u8 addr, int *val) >> { >> + struct st_lsm6dsx_hw *hw = sensor->hw; >> int err, delay; >> __le16 data; >> >> @@ -441,8 +417,7 @@ static int st_lsm6dsx_read_oneshot(struct st_lsm6dsx_sensor *sensor, >> delay = 1000000 / sensor->odr; >> usleep_range(delay, 2 * delay); >> >> - err = sensor->hw->tf->read(sensor->hw->dev, addr, sizeof(data), >> - (u8 *)&data); >> + err = regmap_bulk_read(hw->regmap, addr, &data, sizeof(data)); >> if (err < 0) >> return err; >> >> @@ -657,20 +632,20 @@ static int st_lsm6dsx_get_drdy_reg(struct st_lsm6dsx_hw *hw, u8 *drdy_reg) >> >> static int st_lsm6dsx_init_device(struct st_lsm6dsx_hw *hw) >> { >> - u8 data, drdy_int_reg; >> + u8 drdy_int_reg; >> int err; >> >> - data = ST_LSM6DSX_REG_RESET_MASK; >> - err = hw->tf->write(hw->dev, ST_LSM6DSX_REG_RESET_ADDR, sizeof(data), >> - &data); >> + err = regmap_write(hw->regmap, ST_LSM6DSX_REG_RESET_ADDR, >> + ST_LSM6DSX_REG_RESET_MASK); >> if (err < 0) >> return err; >> >> msleep(200); >> >> /* enable Block Data Update */ >> - err = st_lsm6dsx_write_with_mask(hw, ST_LSM6DSX_REG_BDU_ADDR, >> - ST_LSM6DSX_REG_BDU_MASK, 1); >> + err = regmap_update_bits(hw->regmap, ST_LSM6DSX_REG_BDU_ADDR, >> + ST_LSM6DSX_REG_BDU_MASK, >> + FIELD_PREP(ST_LSM6DSX_REG_BDU_MASK, 1)); >> if (err < 0) >> return err; >> >> @@ -679,8 +654,10 @@ static int st_lsm6dsx_init_device(struct st_lsm6dsx_hw *hw) >> if (err < 0) >> return err; >> >> - return st_lsm6dsx_write_with_mask(hw, drdy_int_reg, >> - ST_LSM6DSX_REG_FIFO_FTH_IRQ_MASK, 1); >> + return regmap_update_bits(hw->regmap, drdy_int_reg, >> + ST_LSM6DSX_REG_FIFO_FTH_IRQ_MASK, >> + FIELD_PREP(ST_LSM6DSX_REG_FIFO_FTH_IRQ_MASK, >> + 1)); >> } >> >> static struct iio_dev *st_lsm6dsx_alloc_iiodev(struct st_lsm6dsx_hw *hw, >> @@ -731,7 +708,7 @@ static struct iio_dev *st_lsm6dsx_alloc_iiodev(struct st_lsm6dsx_hw *hw, >> } >> >> int st_lsm6dsx_probe(struct device *dev, int irq, int hw_id, const char *name, >> - const struct st_lsm6dsx_transfer_function *tf_ops) >> + struct regmap *regmap) >> { >> struct st_lsm6dsx_hw *hw; >> int i, err; >> @@ -742,13 +719,12 @@ int st_lsm6dsx_probe(struct device *dev, int irq, int hw_id, const char *name, >> >> dev_set_drvdata(dev, (void *)hw); >> >> - mutex_init(&hw->lock); >> mutex_init(&hw->fifo_lock); >> mutex_init(&hw->conf_lock); >> >> hw->dev = dev; >> hw->irq = irq; >> - hw->tf = tf_ops; >> + hw->regmap = regmap; >> >> err = st_lsm6dsx_check_whoami(hw, hw_id); >> if (err < 0) >> @@ -784,6 +760,7 @@ static int __maybe_unused st_lsm6dsx_suspend(struct device *dev) >> { >> struct st_lsm6dsx_hw *hw = dev_get_drvdata(dev); >> struct st_lsm6dsx_sensor *sensor; >> + const struct st_lsm6dsx_reg *reg; >> int i, err = 0; >> >> for (i = 0; i < ST_LSM6DSX_ID_MAX; i++) { >> @@ -791,9 +768,9 @@ static int __maybe_unused st_lsm6dsx_suspend(struct device *dev) >> if (!(hw->enable_mask & BIT(sensor->id))) >> continue; >> >> - err = st_lsm6dsx_write_with_mask(hw, >> - st_lsm6dsx_odr_table[sensor->id].reg.addr, >> - st_lsm6dsx_odr_table[sensor->id].reg.mask, 0); >> + reg = &st_lsm6dsx_odr_table[sensor->id].reg; >> + err = regmap_update_bits(hw->regmap, reg->addr, reg->mask, >> + ST_LSM6DSX_SHIFT_VAL(0, reg->mask)); >> if (err < 0) >> return err; >> } >> diff --git a/drivers/iio/imu/st_lsm6dsx/st_lsm6dsx_i2c.c b/drivers/iio/imu/st_lsm6dsx/st_lsm6dsx_i2c.c >> index 305fec712ab0..41525dd2aab7 100644 >> --- a/drivers/iio/imu/st_lsm6dsx/st_lsm6dsx_i2c.c >> +++ b/drivers/iio/imu/st_lsm6dsx/st_lsm6dsx_i2c.c >> @@ -14,55 +14,30 @@ >> #include <linux/i2c.h> >> #include <linux/slab.h> >> #include <linux/of.h> >> +#include <linux/regmap.h> >> >> #include "st_lsm6dsx.h" >> >> -static int st_lsm6dsx_i2c_read(struct device *dev, u8 addr, int len, u8 *data) >> -{ >> - struct i2c_client *client = to_i2c_client(dev); >> - struct i2c_msg msg[2]; >> - >> - msg[0].addr = client->addr; >> - msg[0].flags = client->flags; >> - msg[0].len = 1; >> - msg[0].buf = &addr; >> - >> - msg[1].addr = client->addr; >> - msg[1].flags = client->flags | I2C_M_RD; >> - msg[1].len = len; >> - msg[1].buf = data; >> - >> - return i2c_transfer(client->adapter, msg, 2); >> -} >> - >> -static int st_lsm6dsx_i2c_write(struct device *dev, u8 addr, int len, u8 *data) >> -{ >> - struct i2c_client *client = to_i2c_client(dev); >> - struct i2c_msg msg; >> - u8 send[len + 1]; >> - >> - send[0] = addr; >> - memcpy(&send[1], data, len * sizeof(u8)); >> - >> - msg.addr = client->addr; >> - msg.flags = client->flags; >> - msg.len = len + 1; >> - msg.buf = send; >> - >> - return i2c_transfer(client->adapter, &msg, 1); >> -} >> - >> -static const struct st_lsm6dsx_transfer_function st_lsm6dsx_transfer_fn = { >> - .read = st_lsm6dsx_i2c_read, >> - .write = st_lsm6dsx_i2c_write, >> +static const struct regmap_config st_lsm6dsx_i2c_regmap_config = { >> + .reg_bits = 8, >> + .val_bits = 8, >> }; >> >> static int st_lsm6dsx_i2c_probe(struct i2c_client *client, >> const struct i2c_device_id *id) >> { >> + int hw_id = id->driver_data; >> + struct regmap *regmap; >> + >> + regmap = devm_regmap_init_i2c(client, &st_lsm6dsx_i2c_regmap_config); >> + if (IS_ERR(regmap)) { >> + dev_err(&client->dev, "Failed to register i2c regmap %d\n", >> + (int)PTR_ERR(regmap)); >> + return PTR_ERR(regmap); >> + } >> + >> return st_lsm6dsx_probe(&client->dev, client->irq, >> - (int)id->driver_data, id->name, >> - &st_lsm6dsx_transfer_fn); >> + hw_id, id->name, regmap); >> } >> >> static const struct of_device_id st_lsm6dsx_i2c_of_match[] = { >> diff --git a/drivers/iio/imu/st_lsm6dsx/st_lsm6dsx_spi.c b/drivers/iio/imu/st_lsm6dsx/st_lsm6dsx_spi.c >> index 95472f153ad2..2c8135834479 100644 >> --- a/drivers/iio/imu/st_lsm6dsx/st_lsm6dsx_spi.c >> +++ b/drivers/iio/imu/st_lsm6dsx/st_lsm6dsx_spi.c >> @@ -14,72 +14,30 @@ >> #include <linux/spi/spi.h> >> #include <linux/slab.h> >> #include <linux/of.h> >> +#include <linux/regmap.h> >> >> #include "st_lsm6dsx.h" >> >> -#define SENSORS_SPI_READ BIT(7) >> - >> -static int st_lsm6dsx_spi_read(struct device *dev, u8 addr, int len, >> - u8 *data) >> -{ >> - struct spi_device *spi = to_spi_device(dev); >> - struct st_lsm6dsx_hw *hw = spi_get_drvdata(spi); >> - int err; >> - >> - struct spi_transfer xfers[] = { >> - { >> - .tx_buf = hw->tb.tx_buf, >> - .bits_per_word = 8, >> - .len = 1, >> - }, >> - { >> - .rx_buf = hw->tb.rx_buf, >> - .bits_per_word = 8, >> - .len = len, >> - } >> - }; >> - >> - hw->tb.tx_buf[0] = addr | SENSORS_SPI_READ; >> - >> - err = spi_sync_transfer(spi, xfers, ARRAY_SIZE(xfers)); >> - if (err < 0) >> - return err; >> - >> - memcpy(data, hw->tb.rx_buf, len * sizeof(u8)); >> - >> - return len; >> -} >> - >> -static int st_lsm6dsx_spi_write(struct device *dev, u8 addr, int len, >> - u8 *data) >> -{ >> - struct st_lsm6dsx_hw *hw; >> - struct spi_device *spi; >> - >> - if (len >= ST_LSM6DSX_TX_MAX_LENGTH) >> - return -ENOMEM; >> - >> - spi = to_spi_device(dev); >> - hw = spi_get_drvdata(spi); >> - >> - hw->tb.tx_buf[0] = addr; >> - memcpy(&hw->tb.tx_buf[1], data, len); >> - >> - return spi_write(spi, hw->tb.tx_buf, len + 1); >> -} >> - >> -static const struct st_lsm6dsx_transfer_function st_lsm6dsx_transfer_fn = { >> - .read = st_lsm6dsx_spi_read, >> - .write = st_lsm6dsx_spi_write, >> +static const struct regmap_config st_lsm6dsx_spi_regmap_config = { >> + .reg_bits = 8, >> + .val_bits = 8, >> }; >> >> static int st_lsm6dsx_spi_probe(struct spi_device *spi) >> { >> const struct spi_device_id *id = spi_get_device_id(spi); >> + int hw_id = id->driver_data; >> + struct regmap *regmap; >> + >> + regmap = devm_regmap_init_spi(spi, &st_lsm6dsx_spi_regmap_config); >> + if (IS_ERR(regmap)) { >> + dev_err(&spi->dev, "Failed to register spi regmap %d\n", >> + (int)PTR_ERR(regmap)); >> + return PTR_ERR(regmap); >> + } >> >> return st_lsm6dsx_probe(&spi->dev, spi->irq, >> - (int)id->driver_data, id->name, >> - &st_lsm6dsx_transfer_fn); >> + hw_id, id->name, regmap); >> } >> >> static const struct of_device_id st_lsm6dsx_spi_of_match[] = { > > -- > To unsubscribe from this list: send the line "unsubscribe linux-iio" in > the body of a message to majordomo@xxxxxxxxxxxxxxx > More majordomo info at http://vger.kernel.org/majordomo-info.html -- UNIX is Sexy: who | grep -i blonde | talk; cd ~; wine; talk; touch; unzip; touch; strip; gasp; finger; gasp; mount; fsck; more; yes; gasp; umount; make clean; sleep -- To unsubscribe from this list: send the line "unsubscribe linux-iio" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html