#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; }