coreboot-libre-fam15h-rdimm/3rdparty/chromeec/driver/als_opt3001.c

323 lines
7.2 KiB
C
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/* 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.
*
* TI OPT3001 light sensor driver
*/
#include "common.h"
#include "driver/als_opt3001.h"
#include "i2c.h"
#ifdef HAS_TASK_ALS
/**
* Read register from OPT3001 light sensor.
*/
static int opt3001_i2c_read(const int reg, int *data_ptr)
{
int ret;
ret = i2c_read16(I2C_PORT_ALS, OPT3001_I2C_ADDR_FLAGS,
reg, data_ptr);
if (!ret)
*data_ptr = ((*data_ptr << 8) & 0xFF00) |
((*data_ptr >> 8) & 0x00FF);
return ret;
}
/**
* Write register to OPT3001 light sensor.
*/
static int opt3001_i2c_write(const int reg, int data)
{
data = ((data << 8) & 0xFF00) | ((data >> 8) & 0x00FF);
return i2c_write16(I2C_PORT_ALS, OPT3001_I2C_ADDR_FLAGS,
reg, data);
}
/**
* Initialise OPT3001 light sensor.
*/
int opt3001_init(void)
{
int data;
int ret;
ret = opt3001_i2c_read(OPT3001_REG_MAN_ID, &data);
if (ret)
return ret;
if (data != OPT3001_MANUFACTURER_ID)
return EC_ERROR_UNKNOWN;
ret = opt3001_i2c_read(OPT3001_REG_DEV_ID, &data);
if (ret)
return ret;
if (data != OPT3001_DEVICE_ID)
return EC_ERROR_UNKNOWN;
/*
* [15:12]: 0101b Automatic full scale (1310.40lux, 0.32lux/lsb)
* [11] : 1b Conversion time 800ms
* [10:9] : 10b Continuous Mode of conversion operation
* [4] : 1b Latched window-style comparison operation
*/
return opt3001_i2c_write(OPT3001_REG_CONFIGURE, 0x5C10);
}
/**
* Read OPT3001 light sensor data.
*/
int opt3001_read_lux(int *lux, int af)
{
int ret;
int data;
ret = opt3001_i2c_read(OPT3001_REG_RESULT, &data);
if (ret)
return ret;
/*
* The default power-on values will give 12 bits of precision:
* 0x0000-0x0fff indicates 0 to 1310.40 lux. We multiply the sensor
* value by a scaling factor to account for attenuation by glass,
* tinting, etc.
*/
/*
* lux = 2EXP[3:0] × R[11:0] / 100
*/
*lux = (1 << ((data & 0xF000) >> 12)) * (data & 0x0FFF) * af / 100;
return EC_SUCCESS;
}
#ifdef CONFIG_CMD_I2C_STRESS_TEST_ALS
struct i2c_stress_test_dev opt3001_i2c_stress_test_dev = {
.reg_info = {
.read_reg = OPT3001_REG_DEV_ID,
.read_val = OPT3001_DEVICE_ID,
.write_reg = OPT3001_REG_INT_LIMIT_LSB,
},
.i2c_read_dev = &opt3001_i2c_read,
.i2c_write_dev = &opt3001_i2c_write,
};
#endif /* CONFIG_CMD_I2C_STRESS_TEST_ALS */
#else /* HAS_TASK_ALS */
#include "accelgyro.h"
#include "math_util.h"
/**
* Read register from OPT3001 light sensor.
*/
static int opt3001_i2c_read(const int port,
const uint16_t i2c_addr_flags,
const int reg, int *data_ptr)
{
int ret;
ret = i2c_read16(port, i2c_addr_flags,
reg, data_ptr);
if (!ret)
*data_ptr = ((*data_ptr << 8) & 0xFF00) |
((*data_ptr >> 8) & 0x00FF);
return ret;
}
/**
* Write register to OPT3001 light sensor.
*/
static int opt3001_i2c_write(const int port,
const uint16_t i2c_addr_flags,
const int reg, int data)
{
data = ((data << 8) & 0xFF00) | ((data >> 8) & 0x00FF);
return i2c_write16(port, i2c_addr_flags, reg, data);
}
/**
* Read OPT3001 light sensor data.
*/
int opt3001_read_lux(const struct motion_sensor_t *s, intv3_t v)
{
struct opt3001_drv_data_t *drv_data = OPT3001_GET_DATA(s);
int ret;
int data;
ret = opt3001_i2c_read(s->port, s->i2c_spi_addr_flags,
OPT3001_REG_RESULT, &data);
if (ret)
return ret;
/*
* lux = 2EXP[3:0] × R[11:0] / 100
*/
data = (1 << (data >> 12)) * (data & 0x0FFF);
data += drv_data->offset * 100;
data = data * drv_data->scale + data * drv_data->uscale / 10000;
data /= 100;
if (data < 0)
data = 1;
v[0] = data;
v[1] = 0;
v[2] = 0;
/*
* Return an error when nothing change to prevent filling the
* fifo with useless data.
*/
if (v[0] == drv_data->last_value)
return EC_ERROR_UNCHANGED;
else {
drv_data->last_value = v[0];
return EC_SUCCESS;
}
}
static int opt3001_set_range(const struct motion_sensor_t *s, int range,
int rnd)
{
struct opt3001_drv_data_t *drv_data = OPT3001_GET_DATA(s);
drv_data->scale = range >> 16;
drv_data->uscale = range & 0xffff;
return EC_SUCCESS;
}
static int opt3001_get_range(const struct motion_sensor_t *s)
{
struct opt3001_drv_data_t *drv_data = OPT3001_GET_DATA(s);
return (drv_data->scale << 16) | (drv_data->uscale);
}
static int opt3001_set_data_rate(const struct motion_sensor_t *s,
int rate, int roundup)
{
struct opt3001_drv_data_t *drv_data = OPT3001_GET_DATA(s);
int rv;
int reg;
enum opt3001_mode mode;
if (rate == 0) {
/*
* Suspend driver:
*/
mode = OPT3001_MODE_SUSPEND;
} else {
mode = OPT3001_MODE_CONTINUOUS;
/*
* We set the sensor for continuous mode,
* integrating over 800ms.
* Do not allow range higher than 1Hz.
*/
if (rate > 1000)
rate = 1000;
}
rv = opt3001_i2c_read(s->port, s->i2c_spi_addr_flags,
OPT3001_REG_CONFIGURE, &reg);
if (rv)
return rv;
rv = opt3001_i2c_write(s->port, s->i2c_spi_addr_flags,
OPT3001_REG_CONFIGURE,
(reg & OPT3001_MODE_MASK) |
(mode << OPT3001_MODE_OFFSET));
if (rv)
return rv;
drv_data->rate = rate;
return EC_SUCCESS;
}
static int opt3001_get_data_rate(const struct motion_sensor_t *s)
{
struct opt3001_drv_data_t *drv_data = OPT3001_GET_DATA(s);
return drv_data->rate;
}
static int opt3001_set_offset(const struct motion_sensor_t *s,
const int16_t *offset,
int16_t temp)
{
struct opt3001_drv_data_t *drv_data = OPT3001_GET_DATA(s);
drv_data->offset = offset[X];
return EC_SUCCESS;
}
static int opt3001_get_offset(const struct motion_sensor_t *s,
int16_t *offset,
int16_t *temp)
{
struct opt3001_drv_data_t *drv_data = OPT3001_GET_DATA(s);
offset[X] = drv_data->offset;
offset[Y] = 0;
offset[Z] = 0;
*temp = EC_MOTION_SENSE_INVALID_CALIB_TEMP;
return EC_SUCCESS;
}
/**
* Initialise OPT3001 light sensor.
*/
static int opt3001_init(const struct motion_sensor_t *s)
{
int data;
int ret;
ret = opt3001_i2c_read(s->port, s->i2c_spi_addr_flags,
OPT3001_REG_MAN_ID, &data);
if (ret)
return ret;
if (data != OPT3001_MANUFACTURER_ID)
return EC_ERROR_ACCESS_DENIED;
ret = opt3001_i2c_read(s->port, s->i2c_spi_addr_flags,
OPT3001_REG_DEV_ID, &data);
if (ret)
return ret;
if (data != OPT3001_DEVICE_ID)
return EC_ERROR_ACCESS_DENIED;
/*
* [15-12]: 1100b Automatic full-scale setting mode
* [11] : 1b Conversion time 800ms
* [4] : 1b Latched window-style comparison operation
*/
opt3001_i2c_write(s->port, s->i2c_spi_addr_flags,
OPT3001_REG_CONFIGURE, 0xC810);
opt3001_set_range(s, s->default_range, 0);
return EC_SUCCESS;
}
const struct accelgyro_drv opt3001_drv = {
.init = opt3001_init,
.read = opt3001_read_lux,
.set_range = opt3001_set_range,
.get_range = opt3001_get_range,
.set_offset = opt3001_set_offset,
.get_offset = opt3001_get_offset,
.set_data_rate = opt3001_set_data_rate,
.get_data_rate = opt3001_get_data_rate,
};
#ifdef CONFIG_CMD_I2C_STRESS_TEST_ALS
struct i2c_stress_test_dev opt3001_i2c_stress_test_dev = {
.reg_info = {
.read_reg = OPT3001_REG_DEV_ID,
.read_val = OPT3001_DEVICE_ID,
.write_reg = OPT3001_REG_INT_LIMIT_LSB,
},
.i2c_read = &opt3001_i2c_read,
.i2c_write = &opt3001_i2c_write,
};
#endif /* CONFIG_CMD_I2C_STRESS_TEST_ALS */
#endif /* HAS_TASK_ALS */