Files
OpenCellular/driver/accel_kionix.c
Stefan Reinauer 65bca9b9fd kionix: Initialize rv in all cases
Signed-off-by: Stefan Reinauer <reinauer@chromium.org>
BRANCH=none
BUG=none
TEST=compile tested for Samus

Change-Id: Ib7a0a75a2d63cf8f55d0b59f4a3225da2cb4e70b
Reviewed-on: https://chromium-review.googlesource.com/343282
Commit-Ready: Stefan Reinauer <reinauer@chromium.org>
Tested-by: Stefan Reinauer <reinauer@chromium.org>
Reviewed-by: Patrick Georgi <pgeorgi@chromium.org>
2016-05-09 09:59:42 -07:00

557 lines
14 KiB
C

/* Copyright 2015 The Chromium OS Authors. All rights reserved.
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
/*
* Kionix Accelerometer driver for Chrome EC
*
* Supported: KX022, KXCJ9
*/
#include "accelgyro.h"
#include "common.h"
#include "console.h"
#include "driver/accel_kionix.h"
#include "driver/accel_kx022.h"
#include "driver/accel_kxcj9.h"
#include "i2c.h"
#include "math_util.h"
#include "spi.h"
#include "task.h"
#include "util.h"
#define CPUTS(outstr) cputs(CC_ACCEL, outstr)
#define CPRINTF(format, args...) cprintf(CC_ACCEL, format, ## args)
/* Number of times to attempt to enable sensor before giving up. */
#define SENSOR_ENABLE_ATTEMPTS 3
/* Index for which table to use. */
#if !defined(CONFIG_ACCEL_KXCJ9) || !defined(CONFIG_ACCEL_KX022)
#define T 0
#else
#define T (data->variant)
#endif /* !defined(CONFIG_ACCEL_KXCJ9) || !defined(CONFIG_ACCEL_KX022) */
/* List of range values in +/-G's and their associated register values. */
static const struct accel_param_pair ranges[][3] = {
#ifdef CONFIG_ACCEL_KX022
{ {2, KX022_GSEL_2G},
{4, KX022_GSEL_4G},
{8, KX022_GSEL_8G} },
#endif /* defined(CONFIG_ACCEL_KX022) */
#ifdef CONFIG_ACCEL_KXCJ9
{ {2, KXCJ9_GSEL_2G},
{4, KXCJ9_GSEL_4G},
{8, KXCJ9_GSEL_8G_14BIT} },
#endif /* defined(CONFIG_ACCEL_KXCJ9) */
};
/* List of resolution values in bits and their associated register values. */
static const struct accel_param_pair resolutions[][2] = {
#ifdef CONFIG_ACCEL_KX022
{ {8, KX022_RES_8BIT},
{16, KX022_RES_16BIT} },
#endif /* defined(CONFIG_ACCEL_KX022) */
#ifdef CONFIG_ACCEL_KXCJ9
{ {8, KXCJ9_RES_8BIT},
{12, KXCJ9_RES_12BIT} },
#endif /* defined(CONFIG_ACCEL_KXCJ9) */
};
/* List of ODR values in mHz and their associated register values. */
static const struct accel_param_pair datarates[][13] = {
#ifdef CONFIG_ACCEL_KX022
/* One duplicate because table sizes must match. */
{ {781, KX022_OSA_0_781HZ},
{781, KX022_OSA_0_781HZ},
{1563, KX022_OSA_1_563HZ},
{3125, KX022_OSA_3_125HZ},
{6250, KX022_OSA_6_250HZ},
{12500, KX022_OSA_12_50HZ},
{25000, KX022_OSA_25_00HZ},
{50000, KX022_OSA_50_00HZ},
{100000, KX022_OSA_100_0HZ},
{200000, KX022_OSA_200_0HZ},
{400000, KX022_OSA_400_0HZ},
{800000, KX022_OSA_800_0HZ},
{1600000, KX022_OSA_1600HZ} },
#endif /* defined(CONFIG_ACCEL_KX022) */
#ifdef CONFIG_ACCEL_KXCJ9
{ {0, KXCJ9_OSA_0_000HZ},
{781, KXCJ9_OSA_0_781HZ},
{1563, KXCJ9_OSA_1_563HZ},
{3125, KXCJ9_OSA_3_125HZ},
{6250, KXCJ9_OSA_6_250HZ},
{12500, KXCJ9_OSA_12_50HZ},
{25000, KXCJ9_OSA_25_00HZ},
{50000, KXCJ9_OSA_50_00HZ},
{100000, KXCJ9_OSA_100_0HZ},
{200000, KXCJ9_OSA_200_0HZ},
{400000, KXCJ9_OSA_400_0HZ},
{800000, KXCJ9_OSA_800_0HZ},
{1600000, KXCJ9_OSA_1600_HZ} },
#endif /* defined(CONFIG_ACCEL_KXCJ9) */
};
/**
* Find index into a accel_param_pair that matches the given engineering value
* passed in. The round_up flag is used to specify whether to round up or down.
* Note, this function always returns a valid index. If the request is
* outside the range of values, it returns the closest valid index.
*/
static int find_param_index(const int eng_val, const int round_up,
const struct accel_param_pair *pairs,
const int size)
{
int i;
/* Linear search for index to match. */
for (i = 0; i < size - 1; i++) {
if (eng_val <= pairs[i].val)
return i;
if (eng_val < pairs[i+1].val) {
if (round_up)
return i + 1;
else
return i;
}
}
return i;
}
/**
* Read register from accelerometer.
*/
static int raw_read8(const int port, const int addr, const int reg,
int *data_ptr)
{
int rv = EC_ERROR_INVAL;
if (KIONIX_IS_SPI(addr)) {
#ifdef CONFIG_SPI_ACCEL_PORT
uint8_t val;
uint8_t cmd = 0x80 | reg;
rv = spi_transaction(&spi_devices[KIONIX_SPI_ADDRESS(addr)],
&cmd, 1, &val, 1);
if (rv == EC_SUCCESS)
*data_ptr = val;
#endif
} else {
rv = i2c_read8(port, addr, reg, data_ptr);
}
return rv;
}
/**
* Write register from accelerometer.
*/
static int raw_write8(const int port, const int addr, const int reg, int data)
{
int rv = EC_ERROR_INVAL;
if (KIONIX_IS_SPI(addr)) {
#ifdef CONFIG_SPI_ACCEL_PORT
uint8_t cmd[2] = { reg, data };
rv = spi_transaction(&spi_devices[KIONIX_SPI_ADDRESS(addr)],
cmd, 2, NULL, 0);
#endif
} else {
rv = i2c_write8(port, addr, reg, data);
}
return rv;
}
static int raw_read_multi(const int port, int addr, uint8_t reg,
uint8_t *rxdata, int rxlen)
{
int rv = EC_ERROR_INVAL;
if (KIONIX_IS_SPI(addr)) {
#ifdef CONFIG_SPI_ACCEL_PORT
reg |= 0x80;
rv = spi_transaction(&spi_devices[KIONIX_SPI_ADDRESS(addr)],
&reg, 1, rxdata, rxlen);
#endif
} else {
i2c_lock(port, 1);
rv = i2c_xfer(port, addr, &reg, 1, rxdata, rxlen,
I2C_XFER_SINGLE);
i2c_lock(port, 0);
}
return rv;
}
/**
* Disable sensor by taking it out of operating mode. When disabled, the
* acceleration data does not change.
*
* Note: This is intended to be called in a pair with enable_sensor().
*
* @s Pointer to motion sensor data
* @reg_val Pointer to location to store control register after disabling
*
* @return EC_SUCCESS if successful, EC_ERROR_* otherwise
*/
static int disable_sensor(const struct motion_sensor_t *s, int *reg_val)
{
int i, ret, reg, pc1_field;
struct kionix_accel_data *data = s->drv_data;
reg = KIONIX_CTRL1_REG(data->variant);
pc1_field = KIONIX_PC1_FIELD(data->variant);
/*
* Read the current state of the control register
* so that we can restore it later.
*/
for (i = 0; i < SENSOR_ENABLE_ATTEMPTS; i++) {
ret = raw_read8(s->port, s->addr, reg, reg_val);
if (ret != EC_SUCCESS)
continue;
*reg_val &= ~pc1_field;
ret = raw_write8(s->port, s->addr, reg, *reg_val);
if (ret == EC_SUCCESS)
return EC_SUCCESS;
}
return ret;
}
/**
* Enable sensor by placing it in operating mode.
*
* Note: This is intended to be called in a pair with disable_sensor().
*
* @s Pointer to motion sensor data
* @reg_val Value of the control register to write to sensor
*
* @return EC_SUCCESS if successful, EC_ERROR_* otherwise
*/
static int enable_sensor(const struct motion_sensor_t *s, int reg_val)
{
int i, ret, reg, pc1_field;
struct kionix_accel_data *data = s->drv_data;
reg = KIONIX_CTRL1_REG(data->variant);
pc1_field = KIONIX_PC1_FIELD(data->variant);
for (i = 0; i < SENSOR_ENABLE_ATTEMPTS; i++) {
ret = raw_read8(s->port, s->addr, reg, &reg_val);
if (ret != EC_SUCCESS)
continue;
/* Enable accelerometer based on reg_val value. */
ret = raw_write8(s->port, s->addr, reg,
reg_val | pc1_field);
/* On first success, we are done. */
if (ret == EC_SUCCESS)
return EC_SUCCESS;
}
return ret;
}
static int set_range(const struct motion_sensor_t *s, int range, int rnd)
{
int ret, reg_val, reg_val_new, index, range_field, reg, range_val;
struct kionix_accel_data *data = s->drv_data;
/* Find index for interface pair matching the specified range. */
index = find_param_index(range, rnd, ranges[T], ARRAY_SIZE(ranges[T]));
range_field = KIONIX_RANGE_FIELD(data->variant);
reg = KIONIX_CTRL1_REG(data->variant);
range_val = ranges[T][index].reg;
/* Disable the sensor to allow for changing of critical parameters. */
mutex_lock(s->mutex);
ret = disable_sensor(s, &reg_val);
if (ret != EC_SUCCESS) {
mutex_unlock(s->mutex);
return ret;
}
/* Determine new value of control reg and attempt to write it. */
reg_val_new = (reg_val & ~range_field) | range_val;
ret = raw_write8(s->port, s->addr, reg, reg_val_new);
/* If successfully written, then save the range. */
if (ret == EC_SUCCESS) {
data->sensor_range = index;
reg_val = reg_val_new;
}
/* Re-enable the sensor. */
if (enable_sensor(s, reg_val) != EC_SUCCESS)
ret = EC_ERROR_UNKNOWN;
mutex_unlock(s->mutex);
return ret;
}
static int get_range(const struct motion_sensor_t *s)
{
struct kionix_accel_data *data = s->drv_data;
return ranges[T][data->sensor_range].val;
}
static int set_resolution(const struct motion_sensor_t *s, int res, int rnd)
{
int ret, reg_val, reg_val_new, index, reg, res_field, res_val;
struct kionix_accel_data *data = s->drv_data;
/* Find index for interface pair matching the specified resolution. */
index = find_param_index(res, rnd, resolutions[T],
ARRAY_SIZE(resolutions[T]));
res_val = resolutions[T][index].reg;
res_field = KIONIX_RES_FIELD(data->variant);
reg = KIONIX_CTRL1_REG(data->variant);
/* Disable the sensor to allow for changing of critical parameters. */
mutex_lock(s->mutex);
ret = disable_sensor(s, &reg_val);
if (ret != EC_SUCCESS) {
mutex_unlock(s->mutex);
return ret;
}
/* Determine new value of the control reg and attempt to write it. */
reg_val_new = (reg_val & ~res_field) | res_val;
ret = raw_write8(s->port, s->addr, reg, reg_val_new);
/* If successfully written, then save the range. */
if (ret == EC_SUCCESS) {
data->sensor_resolution = index;
reg_val = reg_val_new;
}
/* Re-enable the sensor. */
if (enable_sensor(s, reg_val) != EC_SUCCESS)
ret = EC_ERROR_UNKNOWN;
mutex_unlock(s->mutex);
return ret;
}
static int get_resolution(const struct motion_sensor_t *s)
{
struct kionix_accel_data *data = s->drv_data;
return resolutions[T][data->sensor_resolution].val;
}
static int set_data_rate(const struct motion_sensor_t *s, int rate, int rnd)
{
int ret, reg_val, index, odr_val, odr_val_new, reg, odr_field,
odr_reg_val;
struct kionix_accel_data *data = s->drv_data;
/* Find index for interface pair matching the specified rate. */
index = find_param_index(rate, rnd, datarates[T],
ARRAY_SIZE(datarates[T]));
odr_val = datarates[T][index].reg;
reg = KIONIX_ODR_REG(data->variant);
odr_field = KIONIX_ODR_FIELD(data->variant);
/* Disable the sensor to allow for changing of critical parameters. */
mutex_lock(s->mutex);
ret = disable_sensor(s, &reg_val);
if (ret != EC_SUCCESS) {
mutex_unlock(s->mutex);
return ret;
}
/* Determine the new value of control reg and attempt to write it. */
ret = raw_read8(s->port, s->addr, reg, &odr_reg_val);
if (ret != EC_SUCCESS) {
mutex_unlock(s->mutex);
return ret;
}
odr_val_new = (odr_reg_val & ~odr_field) | odr_val;
/* Set output data rate. */
ret = raw_write8(s->port, s->addr, reg, odr_val_new);
/* If successfully written, then save the new data rate. */
if (ret == EC_SUCCESS)
data->sensor_datarate = index;
/* Re-enable the sensor. */
if (enable_sensor(s, reg_val) != EC_SUCCESS)
ret = EC_ERROR_UNKNOWN;
mutex_unlock(s->mutex);
return ret;
}
static int get_data_rate(const struct motion_sensor_t *s)
{
struct kionix_accel_data *data = s->drv_data;
return datarates[T][data->sensor_datarate].val;
}
static int set_offset(const struct motion_sensor_t *s, const int16_t *offset,
int16_t temp)
{
/* temperature is ignored */
struct kionix_accel_data *data = s->drv_data;
data->offset[X] = offset[X];
data->offset[Y] = offset[Y];
data->offset[Z] = offset[Z];
return EC_SUCCESS;
}
static int get_offset(const struct motion_sensor_t *s, int16_t *offset,
int16_t *temp)
{
struct kionix_accel_data *data = s->drv_data;
offset[X] = data->offset[X];
offset[Y] = data->offset[Y];
offset[Z] = data->offset[Z];
*temp = EC_MOTION_SENSE_INVALID_CALIB_TEMP;
return EC_SUCCESS;
}
static int read(const struct motion_sensor_t *s, vector_3_t v)
{
uint8_t acc[6];
uint8_t reg;
int ret, i, range, resolution;
struct kionix_accel_data *data = s->drv_data;
/* Read 6 bytes starting at XOUT_L. */
reg = KIONIX_XOUT_L(data->variant);
mutex_lock(s->mutex);
ret = raw_read_multi(s->port, s->addr, reg, acc, 6);
mutex_unlock(s->mutex);
if (ret != EC_SUCCESS)
return ret;
/*
* Convert acceleration to a signed 16-bit number. Note, based on
* the order of the registers:
*
* acc[0] = XOUT_L
* acc[1] = XOUT_H
* acc[2] = YOUT_L
* acc[3] = YOUT_H
* acc[4] = ZOUT_L
* acc[5] = ZOUT_H
*
* Add calibration offset before returning the data.
*/
resolution = get_resolution(s);
for (i = X; i <= Z; i++) {
if (KXCJ9 == data->variant) {
v[i] = (((int8_t)acc[i * 2 + 1]) << 4) |
(acc[i * 2] >> 4);
v[i] <<= 16 - resolution;
} else if (KX022 == data->variant) {
if (resolution == 8)
acc[i * 2] = 0;
v[i] = (((int8_t)acc[i * 2 + 1]) << 8) | acc[i * 2];
}
}
rotate(v, *s->rot_standard_ref, v);
/* apply offset in the device coordinates */
range = get_range(s);
for (i = X; i <= Z; i++)
v[i] += (data->offset[i] << 5) / range;
return EC_SUCCESS;
}
static int init(const struct motion_sensor_t *s)
{
int ret, val, reg, reset_field;
uint8_t timeout;
struct kionix_accel_data *data = s->drv_data;
reg = KIONIX_CTRL2_REG(data->variant);
reset_field = KIONIX_RESET_FIELD(data->variant);
/* Issue a software reset. */
mutex_lock(s->mutex);
/* Place the sensor in standby mode to make changes. */
ret = disable_sensor(s, &val);
if (ret != EC_SUCCESS) {
mutex_unlock(s->mutex);
return ret;
}
ret = raw_read8(s->port, s->addr, reg, &val);
if (ret != EC_SUCCESS) {
mutex_unlock(s->mutex);
return ret;
}
val |= reset_field;
ret = raw_write8(s->port, s->addr, reg, val);
if (ret != EC_SUCCESS) {
mutex_unlock(s->mutex);
return ret;
}
/* The SRST will be cleared when reset is complete. */
timeout = 0;
do {
msleep(1);
ret = raw_read8(s->port, s->addr, reg, &val);
if (ret != EC_SUCCESS) {
mutex_unlock(s->mutex);
return ret;
}
/* Reset complete. */
if ((ret == EC_SUCCESS) && !(val & reset_field))
break;
/* Check for timeout. */
if (timeout++ > 5) {
ret = EC_ERROR_TIMEOUT;
mutex_unlock(s->mutex);
return ret;
}
} while (1);
mutex_unlock(s->mutex);
/* Initialize with the desired parameters. */
ret = set_range(s, s->default_range, 1);
if (ret != EC_SUCCESS)
return ret;
if (KXCJ9 == data->variant)
ret = set_resolution(s, 12, 1);
else if (KX022 == data->variant)
ret = set_resolution(s, 16, 1);
if (ret != EC_SUCCESS)
return ret;
CPRINTF("[%T %s: Done Init type:0x%X range:%d]\n",
s->name, s->type, get_range(s));
mutex_unlock(s->mutex);
return ret;
}
const struct accelgyro_drv kionix_accel_drv = {
.init = init,
.read = read,
.set_range = set_range,
.get_range = get_range,
.set_resolution = set_resolution,
.get_resolution = get_resolution,
.set_data_rate = set_data_rate,
.get_data_rate = get_data_rate,
.set_offset = set_offset,
.get_offset = get_offset,
.perform_calib = NULL,
};