On Sat, 6 Jan 2018 13:44:44 +0100 Lorenzo Bianconi <lorenzo.bianconi83@xxxxxxxxx> wrote: > > 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? sensible option - I'd neglected how large these could get. > > > > > 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? :) More idle wondering than a real suggestions, but take a look at something bitrev.h (grep for __builtin_constant_p to find lots of others. Shows how to do totally different things depending on whether a constant can be evaluated or not. > > 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 > > > -- 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