/*
* Copyright (C) 2016 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include <stdlib.h>
#include <string.h>
#include <float.h>
#include <eventnums.h>
#include <heap.h>
#include <hostIntf.h>
#include <i2c.h>
#include <nanohubPacket.h>
#include <sensors.h>
#include <seos.h>
#include <timer.h>
#define BMP280_APP_ID APP_ID_MAKE(APP_ID_VENDOR_GOOGLE, 5)
#define I2C_BUS_ID 0
#define I2C_SPEED 400000
#define I2C_ADDR 0x76
#define BOSCH_BMP280_ID 0x58
#define BOSCH_BMP280_REG_RESET 0x60
#define BOSCH_BMP280_REG_DIG_T1 0x88
#define BOSCH_BMP280_REG_ID 0xd0
#define BOSCH_BMP280_REG_CTRL_MEAS 0xf4
#define BOSCH_BMP280_REG_CONFIG 0xf5
#define BOSCH_BMP280_REG_PRES_MSB 0xf7
// temp: 2x oversampling, baro: 16x oversampling, power: normal
#define CTRL_ON ((2 << 5) | (5 << 2) | 3)
// temp: 2x oversampling, baro: 16x oversampling, power: sleep
#define CTRL_SLEEP ((2 << 5) | (5 << 2))
enum BMP280SensorEvents
{
EVT_SENSOR_I2C = EVT_APP_START + 1,
EVT_SENSOR_BARO_TIMER,
EVT_SENSOR_TEMP_TIMER,
};
enum BMP280TaskState
{
STATE_RESET,
STATE_VERIFY_ID,
STATE_AWAITING_COMP_PARAMS,
STATE_CONFIG,
STATE_FINISH_INIT,
STATE_IDLE,
STATE_ENABLING_BARO,
STATE_ENABLING_TEMP,
STATE_DISABLING_BARO,
STATE_DISABLING_TEMP,
STATE_SAMPLING,
};
struct BMP280CompParams
{
uint16_t dig_T1;
int16_t dig_T2, dig_T3;
uint16_t dig_P1;
int16_t dig_P2, dig_P3, dig_P4, dig_P5, dig_P6, dig_P7, dig_P8, dig_P9;
} __attribute__((packed));
static struct BMP280Task
{
struct BMP280CompParams comp;
uint32_t id;
uint32_t baroHandle;
uint32_t tempHandle;
uint32_t baroTimerHandle;
uint32_t tempTimerHandle;
float offset;
uint8_t txrxBuf[24];
bool baroOn;
bool tempOn;
bool baroReading;
bool baroCalibrating;
bool tempReading;
} mTask;
struct CalibrationData {
struct HostHubRawPacket header;
struct SensorAppEventHeader data_header;
float value;
} __attribute__((packed));
static const uint32_t tempSupportedRates[] =
{
SENSOR_HZ(0.1),
SENSOR_HZ(1),
SENSOR_HZ(5),
SENSOR_HZ(10),
SENSOR_HZ(25),
0,
};
static const uint64_t rateTimerValsTemp[] = //should match "supported rates in length" and be the timer length for that rate in nanosecs
{
10 * 1000000000ULL,
1 * 1000000000ULL,
1000000000ULL / 5,
1000000000ULL / 10,
1000000000ULL / 25,
};
static const uint32_t baroSupportedRates[] =
{
SENSOR_HZ(0.1),
SENSOR_HZ(1),
SENSOR_HZ(5),
SENSOR_HZ(10),
0
};
static const uint64_t rateTimerValsBaro[] = //should match "supported rates in length" and be the timer length for that rate in nanosecs
{
10 * 1000000000ULL,
1 * 1000000000ULL,
1000000000ULL / 5,
1000000000ULL / 10,
};
/* sensor callbacks from nanohub */
static void i2cCallback(void *cookie, size_t tx, size_t rx, int err)
{
if (err == 0)
osEnqueuePrivateEvt(EVT_SENSOR_I2C, cookie, NULL, mTask.id);
else
osLog(LOG_INFO, "BMP280: i2c error (%d)\n", err);
}
static void baroTimerCallback(uint32_t timerId, void *cookie)
{
osEnqueuePrivateEvt(EVT_SENSOR_BARO_TIMER, cookie, NULL, mTask.id);
}
static void tempTimerCallback(uint32_t timerId, void *cookie)
{
osEnqueuePrivateEvt(EVT_SENSOR_TEMP_TIMER, cookie, NULL, mTask.id);
}
static void setMode(bool on, void *cookie)
{
mTask.txrxBuf[0] = BOSCH_BMP280_REG_CTRL_MEAS;
mTask.txrxBuf[1] = (on) ? CTRL_ON : CTRL_SLEEP;
i2cMasterTx(I2C_BUS_ID, I2C_ADDR, mTask.txrxBuf, 2, &i2cCallback,
cookie);
}
static void sendCalibrationResult(uint8_t status, float value) {
struct CalibrationData *data = heapAlloc(sizeof(struct CalibrationData));
if (!data) {
osLog(LOG_WARN, "Couldn't alloc cal result pkt");
return;
}
data->header.appId = BMP280_APP_ID;
data->header.dataLen = (sizeof(struct CalibrationData) - sizeof(struct HostHubRawPacket));
data->data_header.msgId = SENSOR_APP_MSG_ID_CAL_RESULT;
data->data_header.sensorType = SENS_TYPE_BARO;
data->data_header.status = status;
data->value = value;
if (!osEnqueueEvtOrFree(EVT_APP_TO_HOST, data, heapFree))
osLog(LOG_WARN, "Couldn't send cal result evt");
}
// TODO: only turn on the timer when enabled
static bool sensorPowerBaro(bool on, void *cookie)
{
bool oldMode = mTask.baroOn || mTask.tempOn;
bool newMode = on || mTask.tempOn;
if (!on && mTask.baroTimerHandle) {
timTimerCancel(mTask.baroTimerHandle);
mTask.baroTimerHandle = 0;
mTask.baroReading = false;
}
if (oldMode != newMode)
setMode(newMode, (void*)(on ? STATE_ENABLING_BARO : STATE_DISABLING_BARO));
else
sensorSignalInternalEvt(mTask.baroHandle, SENSOR_INTERNAL_EVT_POWER_STATE_CHG, on, 0);
mTask.baroOn = on;
return true;
}
static bool sensorFirmwareBaro(void *cookie)
{
return sensorSignalInternalEvt(mTask.baroHandle, SENSOR_INTERNAL_EVT_FW_STATE_CHG, 1, 0);
}
static bool sensorRateBaro(uint32_t rate, uint64_t latency, void *cookie)
{
if (mTask.baroTimerHandle)
timTimerCancel(mTask.baroTimerHandle);
mTask.baroTimerHandle = timTimerSet(sensorTimerLookupCommon(baroSupportedRates, rateTimerValsBaro, rate), 0, 50, baroTimerCallback, NULL, false);
return sensorSignalInternalEvt(mTask.baroHandle, SENSOR_INTERNAL_EVT_RATE_CHG, rate, latency);
}
static bool sensorFlushBaro(void *cookie)
{
return osEnqueueEvt(sensorGetMyEventType(SENS_TYPE_BARO), SENSOR_DATA_EVENT_FLUSH, NULL);
}
static bool sensorCalibrateBaro(void *cookie)
{
if (mTask.baroOn || mTask.tempOn) {
osLog(LOG_ERROR, "BMP280: cannot calibrate while baro or temp are active\n");
sendCalibrationResult(SENSOR_APP_EVT_STATUS_BUSY, 0.0f);
return false;
}
if (mTask.baroTimerHandle)
timTimerCancel(mTask.baroTimerHandle);
mTask.baroTimerHandle = timTimerSet(100000000ull, 0, 50, baroTimerCallback, NULL, false);
mTask.offset = 0.0f;
mTask.baroOn = true;
mTask.baroCalibrating = true;
mTask.txrxBuf[0] = BOSCH_BMP280_REG_CTRL_MEAS;
mTask.txrxBuf[1] = CTRL_ON;
i2cMasterTx(I2C_BUS_ID, I2C_ADDR, mTask.txrxBuf, 2, &i2cCallback, (void*)STATE_IDLE);
return true;
}
static bool sensorCfgDataBaro(void *data, void *cookie)
{
mTask.offset = *((float*)data) * 100.0f; // offset is given in hPa, but used as Pa in compensation
return true;
}
static bool sensorPowerTemp(bool on, void *cookie)
{
bool oldMode = mTask.baroOn || mTask.tempOn;
bool newMode = on || mTask.baroOn;
if (!on && mTask.tempTimerHandle) {
timTimerCancel(mTask.tempTimerHandle);
mTask.tempTimerHandle = 0;
mTask.tempReading = false;
}
if (oldMode != newMode)
setMode(newMode, (void*)(on ? STATE_ENABLING_TEMP : STATE_DISABLING_TEMP));
else
sensorSignalInternalEvt(mTask.tempHandle, SENSOR_INTERNAL_EVT_POWER_STATE_CHG, on, 0);
mTask.tempOn = on;
return true;
}
static bool sensorFirmwareTemp(void *cookie)
{
sensorSignalInternalEvt(mTask.tempHandle, SENSOR_INTERNAL_EVT_FW_STATE_CHG, 1, 0);
return true;
}
static bool sensorRateTemp(uint32_t rate, uint64_t latency, void *cookie)
{
if (mTask.tempTimerHandle)
timTimerCancel(mTask.tempTimerHandle);
mTask.tempTimerHandle = timTimerSet(sensorTimerLookupCommon(tempSupportedRates, rateTimerValsTemp, rate), 0, 50, tempTimerCallback, NULL, false);
sensorSignalInternalEvt(mTask.tempHandle, SENSOR_INTERNAL_EVT_RATE_CHG, rate, latency);
return true;
}
static bool sensorFlushTemp(void *cookie)
{
return osEnqueueEvt(sensorGetMyEventType(SENS_TYPE_TEMP), SENSOR_DATA_EVENT_FLUSH, NULL);
}
static const struct SensorInfo sensorInfoBaro =
{
.sensorName = "Pressure",
.supportedRates = baroSupportedRates,
.sensorType = SENS_TYPE_BARO,
.numAxis = NUM_AXIS_EMBEDDED,
.interrupt = NANOHUB_INT_NONWAKEUP,
.minSamples = 300
};
static const struct SensorOps sensorOpsBaro =
{
.sensorPower = sensorPowerBaro,
.sensorFirmwareUpload = sensorFirmwareBaro,
.sensorSetRate = sensorRateBaro,
.sensorFlush = sensorFlushBaro,
.sensorCalibrate = sensorCalibrateBaro,
.sensorCfgData = sensorCfgDataBaro,
};
static const struct SensorInfo sensorInfoTemp =
{
.sensorName = "Temperature",
.supportedRates = tempSupportedRates,
.sensorType = SENS_TYPE_TEMP,
.numAxis = NUM_AXIS_EMBEDDED,
.interrupt = NANOHUB_INT_NONWAKEUP,
.minSamples = 20
};
static const struct SensorOps sensorOpsTemp =
{
.sensorPower = sensorPowerTemp,
.sensorFirmwareUpload = sensorFirmwareTemp,
.sensorSetRate = sensorRateTemp,
.sensorFlush = sensorFlushTemp,
};
// Returns temperature in units of 0.01 degrees celsius.
static int32_t compensateTemp( int32_t adc_T, int32_t *t_fine)
{
int32_t var1 =
(((adc_T >> 3) - ((int32_t)mTask.comp.dig_T1 << 1))
* (int32_t)mTask.comp.dig_T2) >> 11;
int32_t tmp = (adc_T >> 4) - (int32_t)mTask.comp.dig_T1;
int32_t var2 = (((tmp * tmp) >> 12) * (int32_t)mTask.comp.dig_T3) >> 14;
int32_t sum = var1 + var2;
*t_fine = sum;
return (sum * 5 + 128) >> 8;
}
static float compensateBaro(int32_t t_fine, int32_t adc_P)
{
float f = t_fine - 128000, fSqr = f * f;
float a = 1048576 - adc_P;
float v1, v2, p, pSqr;
v2 = fSqr * mTask.comp.dig_P6 + f * mTask.comp.dig_P5 * (float)(1ULL << 17) + mTask.comp.dig_P4 * (float)(1ULL << 35);
v1 = fSqr * mTask.comp.dig_P1 * mTask.comp.dig_P3 * (1.0f/(1ULL << 41)) + f * mTask.comp.dig_P1 * mTask.comp.dig_P2 * (1.0f/(1ULL << 21)) + mTask.comp.dig_P1 * (float)(1ULL << 14);
p = (a * (float)(1ULL << 31) - v2) * 3125 / v1;
pSqr = p * p;
return pSqr * mTask.comp.dig_P9 * (1.0f/(1ULL << 59)) + p * (mTask.comp.dig_P8 * (1.0f/(1ULL << 19)) + 1) * (1.0f/(1ULL << 8)) + 16.0f * mTask.comp.dig_P7;
}
static void getTempAndBaro(const uint8_t *tmp, float *pressure_Pa, float *temp_centigrade)
{
int32_t pres_adc = ((int32_t)tmp[0] << 12) | ((int32_t)tmp[1] << 4) | (tmp[2] >> 4);
int32_t temp_adc = ((int32_t)tmp[3] << 12) | ((int32_t)tmp[4] << 4) | (tmp[5] >> 4);
int32_t T_fine;
int32_t temp = compensateTemp(temp_adc, &T_fine);
float pres = compensateBaro(T_fine, pres_adc);
*temp_centigrade = (float)temp * 0.01f;
*pressure_Pa = pres * (1.0f / 256.0f) + mTask.offset;
}
static void handleI2cEvent(enum BMP280TaskState state)
{
union EmbeddedDataPoint sample;
switch (state) {
case STATE_RESET: {
mTask.txrxBuf[0] = BOSCH_BMP280_REG_ID;
i2cMasterTxRx(I2C_BUS_ID, I2C_ADDR, mTask.txrxBuf, 1,
mTask.txrxBuf, 1, &i2cCallback,
(void*)STATE_VERIFY_ID);
break;
}
case STATE_VERIFY_ID: {
/* Check the sensor ID */
if (mTask.txrxBuf[0] != BOSCH_BMP280_ID) {
osLog(LOG_INFO, "BMP280: not detected\n");
break;
}
/* Get compensation parameters */
mTask.txrxBuf[0] = BOSCH_BMP280_REG_DIG_T1;
i2cMasterTxRx(I2C_BUS_ID, I2C_ADDR, mTask.txrxBuf, 1,
(uint8_t*)&mTask.comp, 24, &i2cCallback,
(void*)STATE_AWAITING_COMP_PARAMS);
break;
}
case STATE_AWAITING_COMP_PARAMS: {
mTask.txrxBuf[0] = BOSCH_BMP280_REG_CTRL_MEAS;
mTask.txrxBuf[1] = CTRL_SLEEP;
i2cMasterTx(I2C_BUS_ID, I2C_ADDR, mTask.txrxBuf, 2,
&i2cCallback, (void*)STATE_CONFIG);
break;
}
case STATE_CONFIG: {
mTask.txrxBuf[0] = BOSCH_BMP280_REG_CONFIG;
// standby time: 62.5ms, IIR filter coefficient: 4
mTask.txrxBuf[1] = (1 << 5) | (2 << 2);
i2cMasterTx(I2C_BUS_ID, I2C_ADDR, mTask.txrxBuf, 2,
&i2cCallback, (void*)STATE_FINISH_INIT);
}
case STATE_ENABLING_BARO: {
sensorSignalInternalEvt(mTask.baroHandle, SENSOR_INTERNAL_EVT_POWER_STATE_CHG, true, 0);
break;
}
case STATE_ENABLING_TEMP: {
sensorSignalInternalEvt(mTask.tempHandle, SENSOR_INTERNAL_EVT_POWER_STATE_CHG, true, 0);
break;
}
case STATE_DISABLING_BARO: {
sensorSignalInternalEvt(mTask.baroHandle, SENSOR_INTERNAL_EVT_POWER_STATE_CHG, false, 0);
break;
}
case STATE_DISABLING_TEMP: {
sensorSignalInternalEvt(mTask.tempHandle, SENSOR_INTERNAL_EVT_POWER_STATE_CHG, false, 0);
break;
}
case STATE_FINISH_INIT: {
sensorRegisterInitComplete(mTask.baroHandle);
sensorRegisterInitComplete(mTask.tempHandle);
osLog(LOG_INFO, "BMP280: idle\n");
break;
}
case STATE_SAMPLING: {
float pressure_Pa, temp_centigrade;
getTempAndBaro(mTask.txrxBuf, &pressure_Pa, &temp_centigrade);
if (mTask.baroOn && mTask.baroReading) {
if (mTask.baroCalibrating) {
sendCalibrationResult(SENSOR_APP_EVT_STATUS_SUCCESS, pressure_Pa * 0.01f);
if (mTask.baroTimerHandle)
timTimerCancel(mTask.baroTimerHandle);
mTask.baroOn = false;
mTask.baroCalibrating = false;
mTask.txrxBuf[0] = BOSCH_BMP280_REG_CTRL_MEAS;
mTask.txrxBuf[1] = CTRL_SLEEP;
i2cMasterTx(I2C_BUS_ID, I2C_ADDR, mTask.txrxBuf, 2, &i2cCallback, (void*)STATE_IDLE);
} else {
sample.fdata = pressure_Pa * 0.01f;
osEnqueueEvt(sensorGetMyEventType(SENS_TYPE_BARO), sample.vptr, NULL);
}
}
if (mTask.tempOn && mTask.tempReading) {
sample.fdata = temp_centigrade;
osEnqueueEvt(sensorGetMyEventType(SENS_TYPE_TEMP), sample.vptr, NULL);
}
mTask.baroReading = false;
mTask.tempReading = false;
break;
}
default:
break;
}
}
static void handleEvent(uint32_t evtType, const void* evtData)
{
switch (evtType) {
case EVT_APP_START:
{
osEventUnsubscribe(mTask.id, EVT_APP_START);
i2cMasterRequest(I2C_BUS_ID, I2C_SPEED);
/* Reset chip */
mTask.txrxBuf[0] = BOSCH_BMP280_REG_RESET;
mTask.txrxBuf[1] = 0xB6;
i2cMasterTx(I2C_BUS_ID, I2C_ADDR, mTask.txrxBuf, 2,
&i2cCallback, (void*)STATE_RESET);
break;
}
case EVT_SENSOR_I2C:
{
handleI2cEvent((enum BMP280TaskState)evtData);
break;
}
case EVT_SENSOR_BARO_TIMER:
{
/* Start sampling for a value */
if (!mTask.baroReading && !mTask.tempReading) {
mTask.txrxBuf[0] = BOSCH_BMP280_REG_PRES_MSB;
i2cMasterTxRx(I2C_BUS_ID, I2C_ADDR, mTask.txrxBuf, 1,
mTask.txrxBuf, 6, &i2cCallback,
(void*)STATE_SAMPLING);
}
mTask.baroReading = true;
break;
}
case EVT_SENSOR_TEMP_TIMER:
{
/* Start sampling for a value */
if (!mTask.baroReading && !mTask.tempReading) {
mTask.txrxBuf[0] = BOSCH_BMP280_REG_PRES_MSB;
i2cMasterTxRx(I2C_BUS_ID, I2C_ADDR, mTask.txrxBuf, 1,
mTask.txrxBuf, 6, &i2cCallback,
(void*)STATE_SAMPLING);
}
mTask.tempReading = true;
break;
}
}
}
static bool startTask(uint32_t taskId)
{
osLog(LOG_INFO, "BMP280: task starting\n");
mTask.id = taskId;
mTask.offset = 0.0f;
/* Register sensors */
mTask.baroHandle = sensorRegister(&sensorInfoBaro, &sensorOpsBaro, NULL, false);
mTask.tempHandle = sensorRegister(&sensorInfoTemp, &sensorOpsTemp, NULL, false);
osEventSubscribe(taskId, EVT_APP_START);
return true;
}
static void endTask(void)
{
}
INTERNAL_APP_INIT(BMP280_APP_ID, 0, startTask, endTask, handleEvent);