blob: 6cee29977d989df21a1a94810f353a9e192e4b54 [file] [log] [blame]
/* Copyright 2016 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.
*/
/**
* Commons acc/gyro function for ST sensors in Chrome EC
*/
#include "stm_mems_common.h"
/**
* st_raw_read_n - Read n bytes for read
*/
int st_raw_read_n(const int port,
const uint16_t i2c_addr_flags,
const uint8_t reg, uint8_t *data_ptr, const int len)
{
/* TODO: Implement SPI interface support */
return i2c_read_block(port, i2c_addr_flags,
reg | 0x80, data_ptr, len);
}
/**
* st_raw_read_n_noinc - Read n bytes for read (no auto inc address)
*/
int st_raw_read_n_noinc(const int port,
const uint16_t i2c_addr_flags,
const uint8_t reg, uint8_t *data_ptr, const int len)
{
/* TODO: Implement SPI interface support */
return i2c_read_block(port, i2c_addr_flags,
reg, data_ptr, len);
}
/**
* st_write_data_with_mask - Write register with mask
* @s: Motion sensor pointer
* @reg: Device register
* @mask: The mask to search
* @data: Data pointer
*/
int st_write_data_with_mask(const struct motion_sensor_t *s, int reg,
uint8_t mask, uint8_t data)
{
int err;
int new_data = 0x00, old_data = 0x00;
err = st_raw_read8(s->port, s->i2c_spi_addr_flags,
reg, &old_data);
if (err != EC_SUCCESS)
return err;
new_data = ((old_data & (~mask)) |
((data << __builtin_ctz(mask)) & mask));
if (new_data == old_data)
return EC_SUCCESS;
return st_raw_write8(s->port, s->i2c_spi_addr_flags,
reg, new_data);
}
/**
* st_get_resolution - Get bit resolution
* @s: Motion sensor pointer
*/
int st_get_resolution(const struct motion_sensor_t *s)
{
struct stprivate_data *data = s->drv_data;
return data->resol;
}
/**
* st_set_offset - Set data offset
* @s: Motion sensor pointer
* @offset: offset vector
* @temp: Temp
*/
int st_set_offset(const struct motion_sensor_t *s,
const int16_t *offset, int16_t temp)
{
struct stprivate_data *data = s->drv_data;
data->offset[X] = offset[X];
data->offset[Y] = offset[Y];
data->offset[Z] = offset[Z];
return EC_SUCCESS;
}
/**
* st_get_offset - Get data offset
* @s: Motion sensor pointer
* @offset: offset vector
* @temp: Temp
*/
int st_get_offset(const struct motion_sensor_t *s,
int16_t *offset, int16_t *temp)
{
struct stprivate_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;
}
/**
* st_get_data_rate - Get data rate (ODR)
* @s: Motion sensor pointer
*/
int st_get_data_rate(const struct motion_sensor_t *s)
{
struct stprivate_data *data = s->drv_data;
return data->base.odr;
}
/**
* st_normalize - Apply LSB data sens. and rotation based on sensor resolution
* @s: Motion sensor pointer
* @v: output vector
* @data: LSB raw data
*/
void st_normalize(const struct motion_sensor_t *s, intv3_t v, uint8_t *data)
{
int i, range;
struct stprivate_data *drvdata = s->drv_data;
/*
* Data is left-aligned and the bottom bits need to be
* cleared because they may contain trash data.
*/
uint16_t mask = ~((1 << (16 - drvdata->resol)) - 1);
for (i = X; i <= Z; i++) {
v[i] = ((data[i * 2 + 1] << 8) | data[i * 2]) & mask;
}
rotate(v, *s->rot_standard_ref, v);
/* apply offset in the device coordinates */
range = s->drv->get_range(s);
for (i = X; i <= Z; i++)
v[i] += (drvdata->offset[i] << 5) / range;
}