Files
mgos_i2c_mock/libs/mgos_barometer/src/mgos_barometer_bme280.c
Pim van Pelt 64fe69d3e3 Introduce 'libs'
Recursively search libs/* for source and include dirs

Move mgos_barometer code to its own subdir, in preparation for release
to GitHub.
2018-04-22 14:04:54 +02:00

143 lines
4.3 KiB
C

#include "mgos_barometer_bme280.h"
#include "mgos_i2c.h"
// Datasheet:
// https://cdn-shop.adafruit.com/datasheets/BST-BME280_DS001-10.pdf
//
// TODO(pim): Add humidity sensing
bool mgos_barometer_bme280_detect(struct mgos_barometer *dev) {
int val;
if (!dev) {
return false;
}
if ((val = mgos_i2c_read_reg_b(dev->i2c, dev->i2caddr, BME280_REG_DEVID)) < 0) {
return false;
}
if (val == 0x56 || val == 0x57) {
LOG(LL_INFO, ("Preproduction version of BMP280 detected (0x%02x)", val));
return true;
}
if (val == 0x58) { // Mass production BMP280
return true;
}
if (val == 0x60) { // Mass production BME280
dev->capabilities |= MGOS_BAROMETER_CAP_HYGROMETER;
return true;
}
return true;
}
bool mgos_barometer_bme280_create(struct mgos_barometer *dev) {
struct mgos_barometer_bme280_data *bme280_data;
if (!dev) {
return false;
}
bme280_data = calloc(1, sizeof(struct mgos_barometer_bme280_data));
if (!bme280_data) {
return false;
}
dev->user_data = bme280_data;
// Reset device
if (!mgos_i2c_write_reg_b(dev->i2c, dev->i2caddr, BME280_REG_RESET, 0xB6)) {
return false;
}
mgos_usleep(10000);
// Read calibration data
if (!mgos_i2c_read_reg_n(dev->i2c, dev->i2caddr, BME280_REG_TEMPERATURE_CALIB_DIG_T1_LSB, 24, (uint8_t *)bme280_data)) {
return false;
}
// SPI | 0.5ms period | 16X IIR filter
if (!mgos_i2c_write_reg_b(dev->i2c, dev->i2caddr, BME280_REG_CONFIG, 0x00 | BME280_STANDBY_500us << 2 | BME280_FILTER_16X << 5)) {
return false;
}
mgos_usleep(10000);
// Mode | Pressure OS | Temp OS
if (!mgos_i2c_write_reg_b(dev->i2c, dev->i2caddr, BME280_REG_CTRL_MEAS, BME280_MODE_NORMAL | BME280_OVERSAMP_16X << 2 | BME280_OVERSAMP_2X << 5)) {
return false;
}
dev->capabilities |= MGOS_BAROMETER_CAP_BAROMETER;
dev->capabilities |= MGOS_BAROMETER_CAP_THERMOMETER;
return true;
}
bool mgos_barometer_bme280_destroy(struct mgos_barometer *dev) {
if (!dev) {
return false;
}
if (dev->user_data) {
free(dev->user_data);
dev->user_data = NULL;
}
return true;
}
bool mgos_barometer_bme280_read(struct mgos_barometer *dev) {
struct mgos_barometer_bme280_data *bme280_data;
if (!dev) {
return false;
}
bme280_data = (struct mgos_barometer_bme280_data *)dev->user_data;
if (!bme280_data) {
return false;
}
// read data from sensor
uint8_t data[6];
if (!mgos_i2c_read_reg_n(dev->i2c, dev->i2caddr, BME280_REG_PRESSURE_MSB, 6, data)) {
return false;
}
int32_t Padc, Tadc;
Padc = (int32_t)((((uint32_t)(data[0])) << 12) | (((uint32_t)(data[1])) << 4) | ((uint32_t)data[2] >> 4));
Tadc = (int32_t)((((uint32_t)(data[3])) << 12) | (((uint32_t)(data[4])) << 4) | ((uint32_t)data[5] >> 4));
// LOG(LL_DEBUG, ("Padc=%d Tadc=%d", Padc, Tadc));
// Convert data (from datasheet, section 8.1)
double var1, var2, T, P;
int32_t t_fine;
// Compensation for temperature -- double precision
var1 = (((double)Tadc) / 16384.0 - ((double)bme280_data->calib.dig_T1) / 1024.0) * ((double)bme280_data->calib.dig_T2);
var2 = ((((double)Tadc) / 131072.0 - ((double)bme280_data->calib.dig_T1) / 8192.0) *
(((double)Tadc) / 131072.0 - ((double)bme280_data->calib.dig_T1) / 8192.0)) * ((double)bme280_data->calib.dig_T3);
t_fine = (int32_t)(var1 + var2);
T = (var1 + var2) / 5120.0;
dev->temperature = (float)T;
// Compensation for pressure -- double precision
var1 = ((double)t_fine / 2.0) - 64000.0;
var2 = var1 * var1 * ((double)bme280_data->calib.dig_P6) / 32768.0;
var2 = var2 + var1 * ((double)bme280_data->calib.dig_P5) * 2.0;
var2 = (var2 / 4.0) + (((double)bme280_data->calib.dig_P4) * 65536.0);
var1 = (((double)bme280_data->calib.dig_P3) * var1 * var1 / 524288.0 + ((double)bme280_data->calib.dig_P2) * var1) / 524288.0;
var1 = (1.0 + var1 / 32768.0) * ((double)bme280_data->calib.dig_P1);
if (var1 == 0.0) {
P = 0.0;
} else {
P = 1048576.0 - (double)Padc;
P = (P - (var2 / 4096.0)) * 6250.0 / var1;
var1 = ((double)bme280_data->calib.dig_P9) * P * P / 2147483648.0;
var2 = P * ((double)bme280_data->calib.dig_P8) / 32768.0;
P = P + (var1 + var2 + ((double)bme280_data->calib.dig_P7)) / 16.0;
}
dev->pressure = (float)P;
// LOG(LL_DEBUG, ("P=%.2f T=%.2f", dev->pressure, dev->temperature));
return true;
}