| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217 |
- // SPDX-License-Identifier: GPL-2.0
- /*
- * Copyright (C) 2019 TDK-InvenSense, Inc.
- */
- #include <linux/kernel.h>
- #include <linux/device.h>
- #include <linux/regmap.h>
- #include <linux/delay.h>
- #include "inv_mpu_aux.h"
- #include "inv_mpu_iio.h"
- /*
- * i2c master auxiliary bus transfer function.
- * Requires the i2c operations to be correctly setup before.
- */
- static int inv_mpu_i2c_master_xfer(const struct inv_mpu6050_state *st)
- {
- /* use 50hz frequency for xfer */
- const unsigned int freq = 50;
- const unsigned int period_ms = 1000 / freq;
- uint8_t d;
- unsigned int user_ctrl;
- int ret;
- /* set sample rate */
- d = INV_MPU6050_FIFO_RATE_TO_DIVIDER(freq);
- ret = regmap_write(st->map, st->reg->sample_rate_div, d);
- if (ret)
- return ret;
- /* start i2c master */
- user_ctrl = st->chip_config.user_ctrl | INV_MPU6050_BIT_I2C_MST_EN;
- ret = regmap_write(st->map, st->reg->user_ctrl, user_ctrl);
- if (ret)
- goto error_restore_rate;
- /* wait for xfer: 1 period + half-period margin */
- msleep(period_ms + period_ms / 2);
- /* stop i2c master */
- user_ctrl = st->chip_config.user_ctrl;
- ret = regmap_write(st->map, st->reg->user_ctrl, user_ctrl);
- if (ret)
- goto error_stop_i2c;
- /* restore sample rate */
- d = st->chip_config.divider;
- ret = regmap_write(st->map, st->reg->sample_rate_div, d);
- if (ret)
- goto error_restore_rate;
- return 0;
- error_stop_i2c:
- regmap_write(st->map, st->reg->user_ctrl, st->chip_config.user_ctrl);
- error_restore_rate:
- regmap_write(st->map, st->reg->sample_rate_div, st->chip_config.divider);
- return ret;
- }
- /**
- * inv_mpu_aux_init() - init i2c auxiliary bus
- * @st: driver internal state
- *
- * Returns 0 on success, a negative error code otherwise.
- */
- int inv_mpu_aux_init(const struct inv_mpu6050_state *st)
- {
- unsigned int val;
- int ret;
- /*
- * Code based on the vendor Linux kernel v3.0,
- * the exact meaning is unknown.
- */
- if (st->chip_type == INV_MPU9150) {
- unsigned int mask = BIT(7);
- val = st->level_shifter ? mask : 0;
- ret = regmap_update_bits(st->map, 0x1, mask, val);
- if (ret)
- return ret;
- }
- /* configure i2c master */
- val = INV_MPU6050_BITS_I2C_MST_CLK_400KHZ |
- INV_MPU6050_BIT_WAIT_FOR_ES;
- ret = regmap_write(st->map, INV_MPU6050_REG_I2C_MST_CTRL, val);
- if (ret)
- return ret;
- /* configure i2c master delay */
- ret = regmap_write(st->map, INV_MPU6050_REG_I2C_SLV4_CTRL, 0);
- if (ret)
- return ret;
- val = INV_MPU6050_BIT_I2C_SLV0_DLY_EN |
- INV_MPU6050_BIT_I2C_SLV1_DLY_EN |
- INV_MPU6050_BIT_I2C_SLV2_DLY_EN |
- INV_MPU6050_BIT_I2C_SLV3_DLY_EN |
- INV_MPU6050_BIT_DELAY_ES_SHADOW;
- return regmap_write(st->map, INV_MPU6050_REG_I2C_MST_DELAY_CTRL, val);
- }
- /**
- * inv_mpu_aux_read() - read register function for i2c auxiliary bus
- * @st: driver internal state.
- * @addr: chip i2c Address
- * @reg: chip register address
- * @val: buffer for storing read bytes
- * @size: number of bytes to read
- *
- * Returns 0 on success, a negative error code otherwise.
- */
- int inv_mpu_aux_read(const struct inv_mpu6050_state *st, uint8_t addr,
- uint8_t reg, uint8_t *val, size_t size)
- {
- unsigned int status;
- int ret;
- if (size > 0x0F)
- return -EINVAL;
- /* setup i2c SLV0 control: i2c addr, register, enable + size */
- ret = regmap_write(st->map, INV_MPU6050_REG_I2C_SLV_ADDR(0),
- INV_MPU6050_BIT_I2C_SLV_RNW | addr);
- if (ret)
- return ret;
- ret = regmap_write(st->map, INV_MPU6050_REG_I2C_SLV_REG(0), reg);
- if (ret)
- return ret;
- ret = regmap_write(st->map, INV_MPU6050_REG_I2C_SLV_CTRL(0),
- INV_MPU6050_BIT_SLV_EN | size);
- if (ret)
- return ret;
- /* do i2c xfer */
- ret = inv_mpu_i2c_master_xfer(st);
- if (ret)
- goto error_disable_i2c;
- /* disable i2c slave */
- ret = regmap_write(st->map, INV_MPU6050_REG_I2C_SLV_CTRL(0), 0);
- if (ret)
- goto error_disable_i2c;
- /* check i2c status */
- ret = regmap_read(st->map, INV_MPU6050_REG_I2C_MST_STATUS, &status);
- if (ret)
- return ret;
- if (status & INV_MPU6050_BIT_I2C_SLV0_NACK)
- return -EIO;
- /* read data in registers */
- return regmap_bulk_read(st->map, INV_MPU6050_REG_EXT_SENS_DATA,
- val, size);
- error_disable_i2c:
- regmap_write(st->map, INV_MPU6050_REG_I2C_SLV_CTRL(0), 0);
- return ret;
- }
- /**
- * inv_mpu_aux_write() - write register function for i2c auxiliary bus
- * @st: driver internal state.
- * @addr: chip i2c Address
- * @reg: chip register address
- * @val: 1 byte value to write
- *
- * Returns 0 on success, a negative error code otherwise.
- */
- int inv_mpu_aux_write(const struct inv_mpu6050_state *st, uint8_t addr,
- uint8_t reg, uint8_t val)
- {
- unsigned int status;
- int ret;
- /* setup i2c SLV0 control: i2c addr, register, value, enable + size */
- ret = regmap_write(st->map, INV_MPU6050_REG_I2C_SLV_ADDR(0), addr);
- if (ret)
- return ret;
- ret = regmap_write(st->map, INV_MPU6050_REG_I2C_SLV_REG(0), reg);
- if (ret)
- return ret;
- ret = regmap_write(st->map, INV_MPU6050_REG_I2C_SLV_DO(0), val);
- if (ret)
- return ret;
- ret = regmap_write(st->map, INV_MPU6050_REG_I2C_SLV_CTRL(0),
- INV_MPU6050_BIT_SLV_EN | 1);
- if (ret)
- return ret;
- /* do i2c xfer */
- ret = inv_mpu_i2c_master_xfer(st);
- if (ret)
- goto error_disable_i2c;
- /* disable i2c slave */
- ret = regmap_write(st->map, INV_MPU6050_REG_I2C_SLV_CTRL(0), 0);
- if (ret)
- goto error_disable_i2c;
- /* check i2c status */
- ret = regmap_read(st->map, INV_MPU6050_REG_I2C_MST_STATUS, &status);
- if (ret)
- return ret;
- if (status & INV_MPU6050_BIT_I2C_SLV0_NACK)
- return -EIO;
- return 0;
- error_disable_i2c:
- regmap_write(st->map, INV_MPU6050_REG_I2C_SLV_CTRL(0), 0);
- return ret;
- }
|