2019-03-13 21:15:28 +01:00
|
|
|
#if (HAS_BME)
|
2018-11-22 23:37:53 +01:00
|
|
|
|
2019-03-13 21:15:28 +01:00
|
|
|
#include "bmesensor.h"
|
2018-11-22 23:37:53 +01:00
|
|
|
|
|
|
|
// Local logging tag
|
2019-02-27 00:49:32 +01:00
|
|
|
static const char TAG[] = __FILE__;
|
2018-11-22 23:37:53 +01:00
|
|
|
|
2019-07-29 20:23:27 +02:00
|
|
|
bmeStatus_t bme_status = {0};
|
|
|
|
|
|
|
|
Ticker bmecycler;
|
2018-11-26 16:12:41 +01:00
|
|
|
|
2019-03-13 21:15:28 +01:00
|
|
|
#define SEALEVELPRESSURE_HPA (1013.25)
|
|
|
|
|
|
|
|
#ifdef HAS_BME680
|
2018-12-27 17:09:40 +01:00
|
|
|
bsec_virtual_sensor_t sensorList[10] = {
|
|
|
|
BSEC_OUTPUT_RAW_TEMPERATURE,
|
|
|
|
BSEC_OUTPUT_RAW_PRESSURE,
|
|
|
|
BSEC_OUTPUT_RAW_HUMIDITY,
|
|
|
|
BSEC_OUTPUT_RAW_GAS,
|
|
|
|
BSEC_OUTPUT_IAQ,
|
|
|
|
BSEC_OUTPUT_STATIC_IAQ,
|
|
|
|
BSEC_OUTPUT_CO2_EQUIVALENT,
|
|
|
|
BSEC_OUTPUT_BREATH_VOC_EQUIVALENT,
|
|
|
|
BSEC_OUTPUT_SENSOR_HEAT_COMPENSATED_TEMPERATURE,
|
|
|
|
BSEC_OUTPUT_SENSOR_HEAT_COMPENSATED_HUMIDITY,
|
|
|
|
};
|
|
|
|
|
2019-01-06 19:41:42 +01:00
|
|
|
uint8_t bsecstate_buffer[BSEC_MAX_STATE_BLOB_SIZE] = {0};
|
2018-12-31 11:26:05 +01:00
|
|
|
uint16_t stateUpdateCounter = 0;
|
|
|
|
|
2019-01-06 19:41:42 +01:00
|
|
|
Bsec iaqSensor;
|
|
|
|
|
2019-03-13 21:15:28 +01:00
|
|
|
#elif defined HAS_BME280
|
|
|
|
|
2019-12-15 17:03:33 +01:00
|
|
|
Adafruit_BME280 bme; // using I2C interface
|
|
|
|
|
|
|
|
// use these alternative constructors for other hw interface types
|
|
|
|
// Adafruit_BME280 bme(BME_CS); // hardware SPI
|
2019-03-13 21:15:28 +01:00
|
|
|
// Adafruit_BME280 bme(BME_CS, BME_MOSI, BME_MISO, BME_SCK); // software SPI
|
|
|
|
|
2019-12-08 12:58:12 +01:00
|
|
|
#elif defined HAS_BMP180
|
|
|
|
|
|
|
|
Adafruit_BMP085 bmp; // I2C
|
|
|
|
|
2019-03-13 21:15:28 +01:00
|
|
|
#endif
|
|
|
|
|
2020-10-04 14:29:57 +02:00
|
|
|
void setBMEIRQ() { xTaskNotify(irqHandlerTask, BME_IRQ, eSetBits); }
|
2019-07-29 20:23:27 +02:00
|
|
|
|
2020-10-30 12:25:15 +01:00
|
|
|
// initialize MEMS sensor
|
2018-11-25 12:49:49 +01:00
|
|
|
int bme_init(void) {
|
2018-12-31 15:46:52 +01:00
|
|
|
|
|
|
|
// return = 0 -> error / return = 1 -> success
|
2019-07-25 22:04:38 +02:00
|
|
|
int rc = 1;
|
2018-12-31 15:46:52 +01:00
|
|
|
|
2019-03-13 21:15:28 +01:00
|
|
|
#ifdef HAS_BME680
|
2018-12-27 17:09:40 +01:00
|
|
|
// block i2c bus access
|
2019-01-26 12:32:17 +01:00
|
|
|
if (I2C_MUTEX_LOCK()) {
|
2018-11-22 23:37:53 +01:00
|
|
|
|
2019-03-08 18:13:51 +01:00
|
|
|
Wire.begin(HAS_BME680);
|
|
|
|
iaqSensor.begin(BME680_ADDR, Wire);
|
2018-11-25 16:05:30 +01:00
|
|
|
|
2018-12-27 17:09:40 +01:00
|
|
|
ESP_LOGI(TAG, "BSEC v%d.%d.%d.%d", iaqSensor.version.major,
|
|
|
|
iaqSensor.version.minor, iaqSensor.version.major_bugfix,
|
|
|
|
iaqSensor.version.minor_bugfix);
|
2018-12-21 00:35:21 +01:00
|
|
|
|
2018-12-27 17:09:40 +01:00
|
|
|
iaqSensor.setConfig(bsec_config_iaq);
|
2018-11-22 23:37:53 +01:00
|
|
|
|
2018-12-27 17:09:40 +01:00
|
|
|
if (checkIaqSensorStatus())
|
|
|
|
ESP_LOGI(TAG, "BME680 sensor found and initialized");
|
|
|
|
else {
|
|
|
|
ESP_LOGE(TAG, "BME680 sensor not found");
|
2019-07-25 22:04:38 +02:00
|
|
|
rc = 0;
|
|
|
|
goto finish;
|
2018-12-27 17:09:40 +01:00
|
|
|
}
|
2018-11-25 11:48:03 +01:00
|
|
|
|
2018-12-31 11:26:05 +01:00
|
|
|
loadState();
|
|
|
|
|
2018-12-27 17:09:40 +01:00
|
|
|
iaqSensor.setTemperatureOffset((float)BME_TEMP_OFFSET);
|
|
|
|
iaqSensor.updateSubscription(sensorList, 10, BSEC_SAMPLE_RATE_LP);
|
2018-11-22 23:37:53 +01:00
|
|
|
|
2018-12-27 17:09:40 +01:00
|
|
|
if (checkIaqSensorStatus())
|
|
|
|
ESP_LOGI(TAG, "BSEC subscription succesful");
|
|
|
|
else {
|
|
|
|
ESP_LOGE(TAG, "BSEC subscription error");
|
2019-07-25 22:04:38 +02:00
|
|
|
rc = 0;
|
|
|
|
goto finish;
|
2018-12-27 17:09:40 +01:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
ESP_LOGE(TAG, "I2c bus busy - BME680 initialization error");
|
2019-07-25 22:04:38 +02:00
|
|
|
rc = 0;
|
|
|
|
goto finish;
|
2018-11-22 23:37:53 +01:00
|
|
|
}
|
2018-12-27 17:09:40 +01:00
|
|
|
|
2019-03-13 21:15:28 +01:00
|
|
|
#elif defined HAS_BME280
|
|
|
|
|
|
|
|
bool status;
|
|
|
|
|
|
|
|
// block i2c bus access
|
|
|
|
if (I2C_MUTEX_LOCK()) {
|
2019-07-25 22:04:38 +02:00
|
|
|
|
2019-03-13 21:15:28 +01:00
|
|
|
status = bme.begin(BME280_ADDR);
|
|
|
|
if (!status) {
|
|
|
|
ESP_LOGE(TAG, "BME280 sensor not found");
|
2019-07-25 22:04:38 +02:00
|
|
|
rc = 0;
|
|
|
|
goto finish;
|
2019-03-13 21:15:28 +01:00
|
|
|
}
|
|
|
|
ESP_LOGI(TAG, "BME280 sensor found and initialized");
|
|
|
|
} else {
|
|
|
|
ESP_LOGE(TAG, "I2c bus busy - BME280 initialization error");
|
2019-07-25 22:04:38 +02:00
|
|
|
rc = 0;
|
|
|
|
goto finish;
|
2019-03-13 21:15:28 +01:00
|
|
|
}
|
|
|
|
|
2019-12-08 12:58:12 +01:00
|
|
|
#elif defined HAS_BMP180
|
|
|
|
bool status;
|
|
|
|
// block i2c bus access
|
|
|
|
if (I2C_MUTEX_LOCK()) {
|
|
|
|
// Wire.begin(21, 22);
|
|
|
|
status = bmp.begin();
|
|
|
|
if (!status) {
|
|
|
|
ESP_LOGE(TAG, "BMP180 sensor not found");
|
|
|
|
rc = 0;
|
|
|
|
goto finish;
|
|
|
|
}
|
|
|
|
ESP_LOGI(TAG, "BMP180 sensor found and initialized");
|
|
|
|
} else {
|
|
|
|
ESP_LOGE(TAG, "I2c bus busy - BMP180 initialization error");
|
|
|
|
rc = 0;
|
|
|
|
goto finish;
|
|
|
|
}
|
2019-03-13 21:15:28 +01:00
|
|
|
#endif
|
|
|
|
|
2019-07-25 22:04:38 +02:00
|
|
|
finish:
|
2019-01-26 12:32:17 +01:00
|
|
|
I2C_MUTEX_UNLOCK(); // release i2c bus access
|
2019-07-31 22:20:18 +02:00
|
|
|
if (rc)
|
2020-10-04 14:29:57 +02:00
|
|
|
bmecycler.attach(BMECYCLE, setBMEIRQ);
|
2019-07-25 22:04:38 +02:00
|
|
|
return rc;
|
2018-12-31 15:46:52 +01:00
|
|
|
|
2018-12-27 17:09:40 +01:00
|
|
|
} // bme_init()
|
2018-11-22 23:37:53 +01:00
|
|
|
|
2019-03-13 21:15:28 +01:00
|
|
|
#ifdef HAS_BME680
|
|
|
|
|
2018-12-26 23:01:54 +01:00
|
|
|
// Helper function definitions
|
|
|
|
int checkIaqSensorStatus(void) {
|
|
|
|
int rslt = 1; // true = 1 = no error, false = 0 = error
|
|
|
|
|
|
|
|
if (iaqSensor.status != BSEC_OK) {
|
|
|
|
rslt = 0;
|
|
|
|
if (iaqSensor.status < BSEC_OK)
|
|
|
|
ESP_LOGE(TAG, "BSEC error %d", iaqSensor.status);
|
|
|
|
else
|
|
|
|
ESP_LOGW(TAG, "BSEC warning %d", iaqSensor.status);
|
2018-11-22 23:37:53 +01:00
|
|
|
}
|
|
|
|
|
2018-12-26 23:01:54 +01:00
|
|
|
if (iaqSensor.bme680Status != BME680_OK) {
|
|
|
|
rslt = 0;
|
|
|
|
if (iaqSensor.bme680Status < BME680_OK)
|
|
|
|
ESP_LOGE(TAG, "BME680 error %d", iaqSensor.bme680Status);
|
|
|
|
else
|
|
|
|
ESP_LOGW(TAG, "BME680 warning %d", iaqSensor.bme680Status);
|
|
|
|
}
|
2018-11-25 11:48:03 +01:00
|
|
|
|
2018-12-26 23:01:54 +01:00
|
|
|
return rslt;
|
2018-12-27 17:09:40 +01:00
|
|
|
} // checkIaqSensorStatus()
|
2019-03-13 21:15:28 +01:00
|
|
|
#endif
|
2018-11-25 11:48:03 +01:00
|
|
|
|
2019-07-29 20:23:27 +02:00
|
|
|
// store current BME sensor data in struct
|
|
|
|
void bme_storedata(bmeStatus_t *bme_store) {
|
2019-08-25 17:18:29 +02:00
|
|
|
|
2019-09-01 00:11:20 +02:00
|
|
|
if ((cfg.payloadmask & MEMS_DATA) &&
|
2019-08-25 17:18:29 +02:00
|
|
|
(I2C_MUTEX_LOCK())) { // block i2c bus access
|
2018-11-25 11:48:03 +01:00
|
|
|
|
2019-03-08 18:13:51 +01:00
|
|
|
#ifdef HAS_BME680
|
2019-07-29 20:23:27 +02:00
|
|
|
if (iaqSensor.run()) { // if new data is available
|
|
|
|
bme_store->raw_temperature =
|
|
|
|
iaqSensor.rawTemperature; // temperature in degree celsius
|
|
|
|
bme_store->raw_humidity = iaqSensor.rawHumidity;
|
|
|
|
bme_store->temperature = iaqSensor.temperature;
|
|
|
|
bme_store->humidity =
|
|
|
|
iaqSensor.humidity; // humidity in % relative humidity x1000
|
|
|
|
bme_store->pressure = // pressure in Pascal
|
|
|
|
(iaqSensor.pressure / 100.0); // conversion Pa -> hPa
|
2020-01-28 21:49:24 +01:00
|
|
|
bme_store->iaq = iaqSensor.iaq;
|
2019-07-29 20:23:27 +02:00
|
|
|
bme_store->iaq_accuracy = iaqSensor.iaqAccuracy;
|
|
|
|
bme_store->gas = iaqSensor.gasResistance; // gas resistance in ohms
|
|
|
|
updateState();
|
2018-12-26 23:01:54 +01:00
|
|
|
}
|
2019-03-13 21:15:28 +01:00
|
|
|
#elif defined HAS_BME280
|
2019-07-29 20:23:27 +02:00
|
|
|
bme_store->temperature = bme.readTemperature();
|
|
|
|
bme_store->pressure = (bme.readPressure() / 100.0); // conversion Pa -> hPa
|
|
|
|
// bme.readAltitude(SEALEVELPRESSURE_HPA);
|
|
|
|
bme_store->humidity = bme.readHumidity();
|
|
|
|
bme_store->iaq = 0; // IAQ feature not present with BME280
|
2019-12-08 12:58:12 +01:00
|
|
|
#elif defined HAS_BMP180
|
|
|
|
bme_store->temperature = bmp.readTemperature();
|
|
|
|
bme_store->pressure = (bmp.readPressure() / 100.0); // conversion Pa -> hPa
|
|
|
|
// bme.readAltitude(SEALEVELPRESSURE_HPA);
|
|
|
|
bme_store->iaq = 0; // IAQ feature not present with BME280
|
2018-12-26 23:01:54 +01:00
|
|
|
#endif
|
2018-11-22 23:37:53 +01:00
|
|
|
|
2019-07-29 20:23:27 +02:00
|
|
|
I2C_MUTEX_UNLOCK(); // release i2c bus access
|
|
|
|
}
|
|
|
|
|
|
|
|
} // bme_storedata()
|
2018-11-25 11:48:03 +01:00
|
|
|
|
2019-03-13 21:15:28 +01:00
|
|
|
#ifdef HAS_BME680
|
2018-12-31 11:26:05 +01:00
|
|
|
void loadState(void) {
|
2019-01-01 21:23:45 +01:00
|
|
|
if (cfg.bsecstate[BSEC_MAX_STATE_BLOB_SIZE] == BSEC_MAX_STATE_BLOB_SIZE) {
|
2018-12-31 11:26:05 +01:00
|
|
|
// Existing state in NVS stored
|
|
|
|
ESP_LOGI(TAG, "restoring BSEC state from NVRAM");
|
2018-12-31 15:46:52 +01:00
|
|
|
memcpy(bsecstate_buffer, cfg.bsecstate, BSEC_MAX_STATE_BLOB_SIZE);
|
|
|
|
iaqSensor.setState(bsecstate_buffer);
|
2018-12-31 11:26:05 +01:00
|
|
|
checkIaqSensorStatus();
|
|
|
|
} else // no state stored
|
|
|
|
ESP_LOGI(TAG,
|
|
|
|
"no BSEC state stored in NVRAM, starting sensor with defaults");
|
|
|
|
}
|
|
|
|
|
|
|
|
void updateState(void) {
|
|
|
|
bool update = false;
|
|
|
|
|
|
|
|
if (stateUpdateCounter == 0) {
|
2019-01-06 19:41:42 +01:00
|
|
|
// first state update when IAQ accuracy is >= 1
|
|
|
|
if (iaqSensor.iaqAccuracy >= 1) {
|
2018-12-31 11:26:05 +01:00
|
|
|
update = true;
|
|
|
|
stateUpdateCounter++;
|
|
|
|
}
|
|
|
|
} else {
|
2019-01-01 21:23:45 +01:00
|
|
|
|
2018-12-31 11:26:05 +01:00
|
|
|
/* Update every STATE_SAVE_PERIOD minutes */
|
2020-11-02 12:39:46 +01:00
|
|
|
if ((stateUpdateCounter * STATE_SAVE_PERIOD) < _millis()) {
|
2018-12-31 11:26:05 +01:00
|
|
|
update = true;
|
|
|
|
stateUpdateCounter++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (update) {
|
2018-12-31 15:46:52 +01:00
|
|
|
iaqSensor.getState(bsecstate_buffer);
|
2018-12-31 11:26:05 +01:00
|
|
|
checkIaqSensorStatus();
|
2019-01-01 14:59:00 +01:00
|
|
|
memcpy(cfg.bsecstate, bsecstate_buffer, BSEC_MAX_STATE_BLOB_SIZE);
|
2019-01-01 21:23:45 +01:00
|
|
|
cfg.bsecstate[BSEC_MAX_STATE_BLOB_SIZE] = BSEC_MAX_STATE_BLOB_SIZE;
|
2018-12-31 11:26:05 +01:00
|
|
|
ESP_LOGI(TAG, "saving BSEC state to NVRAM");
|
|
|
|
saveConfig();
|
|
|
|
}
|
|
|
|
}
|
2019-03-13 21:15:28 +01:00
|
|
|
#endif
|
2018-12-31 11:26:05 +01:00
|
|
|
|
2019-03-13 21:15:28 +01:00
|
|
|
#endif // HAS_BME
|