Recursively search libs/* for source and include dirs Move mgos_barometer code to its own subdir, in preparation for release to GitHub.
143 lines
4.3 KiB
C
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;
|
|
}
|