10#ifndef M5_UNIT_ENV_UNIT_BME688_HPP
11#define M5_UNIT_ENV_UNIT_BME688_HPP
13#include <M5UnitComponent.hpp>
14#include <m5_utility/stl/extension.hpp>
17#include <bme68xLibrary.h>
19#include <bme68x/bme68x.h>
22#if defined(CONFIG_IDF_TARGET_ESP32) || defined(CONFIG_IDF_TARGET_ESP32S3) || defined(CONFIG_IDF_TARGET_ESP32C3)
23#define UNIT_BME688_USING_BSEC2
28#include <inc/bsec_datatypes.h>
35#include <initializer_list>
77 uint16_t temp_prof[10]{};
78 uint16_t dur_prof[10]{};
81 heatr_temp_prof = temp_prof;
82 heatr_dur_prof = dur_prof;
152 inline uint8_t step()
const
156 inline Factor factor()
const
164 inline void step(
const uint8_t s)
168 inline void factor(
const Factor f)
170 value = (
value & ~(0x03 << 6)) | (m5::stl::to_underlying(f) << 6);
176 static uint8_t
from(
const uint16_t duration)
179 uint16_t d{duration};
184 return (f <= 0x03) ? ((uint8_t)d | (f << 6)) : 0xFF;
188 static uint16_t
to(
const uint8_t v)
190 constexpr uint16_t tbl[] = {1, 4, 16, 64};
191 return (v & 0x3F) * tbl[(v >> 6) & 0x03];
197#if defined(UNIT_BME688_USING_BSEC2)
204enum class SampleRate : uint8_t {
208 UltraLowPowerMeasurementOnDemand,
216void is_bsec_virtual_sensor_t()
218 static_assert(std::is_same<T, bsec_virtual_sensor_t>::value,
"Argument must be of type bsec_virtual_sensor_t");
223inline uint32_t virtual_sensor_array_to_bits(
const bsec_virtual_sensor_t* ss,
const size_t len)
226 for (
size_t i = 0; i < len; ++i) {
227 ret |= ((uint32_t)1U) << ss[i];
235template <
typename... Args>
236uint32_t subscribe_to_bits(Args... args)
241 int discard[] = {(is_bsec_virtual_sensor_t<Args>(), 0)...};
244 bsec_virtual_sensor_t tmp[] = {args...};
245 constexpr size_t n =
sizeof...(args);
246 return virtual_sensor_array_to_bits(tmp, n);
258#if defined(UNIT_BME688_USING_BSEC2)
259 bsecOutputs raw_outputs{};
261 float get(
const bsec_virtual_sensor_t vs)
const;
262 inline float iaq()
const
264 return get(BSEC_OUTPUT_IAQ);
266 inline float static_iaq()
const
268 return get(BSEC_OUTPUT_STATIC_IAQ);
270 inline float co2()
const
272 return get(BSEC_OUTPUT_CO2_EQUIVALENT);
274 inline float voc()
const
276 return get(BSEC_OUTPUT_BREATH_VOC_EQUIVALENT);
278 inline float temperature()
const
280 return get(BSEC_OUTPUT_RAW_TEMPERATURE);
282 inline float pressure()
const
284 return get(BSEC_OUTPUT_RAW_PRESSURE);
286 inline float humidity()
const
288 return get(BSEC_OUTPUT_RAW_HUMIDITY);
290 inline float gas()
const
292 return get(BSEC_OUTPUT_RAW_GAS);
294 inline bool gas_stabilization()
const
296 return get(BSEC_OUTPUT_STABILIZATION_STATUS) == 1.0f;
298 inline bool gas_run_in_status()
const
300 return get(BSEC_OUTPUT_RUN_IN_STATUS) == 1.0f;
302 inline float heat_compensated_temperature()
const
304 return get(BSEC_OUTPUT_SENSOR_HEAT_COMPENSATED_TEMPERATURE);
306 inline float heat_compensated_humidity()
const
308 return get(BSEC_OUTPUT_SENSOR_HEAT_COMPENSATED_HUMIDITY);
310 inline float gas_percentage()
const
312 return get(BSEC_OUTPUT_GAS_PERCENTAGE);
314 inline float gas_estimate_1()
const
316 return get(BSEC_OUTPUT_GAS_ESTIMATE_1);
318 inline float gas_estimate_2()
const
320 return get(BSEC_OUTPUT_GAS_ESTIMATE_2);
322 inline float gas_estimate_3()
const
324 return get(BSEC_OUTPUT_GAS_ESTIMATE_3);
326 inline float gas_estimate_4()
const
328 return get(BSEC_OUTPUT_GAS_ESTIMATE_4);
330 inline uint32_t gas_index()
const
332 return get(BSEC_OUTPUT_RAW_GAS_INDEX);
334 inline float regression_estimate_1()
const
336 return get(BSEC_OUTPUT_REGRESSION_ESTIMATE_1);
338 inline float regression_estimate_2()
const
340 return get(BSEC_OUTPUT_REGRESSION_ESTIMATE_2);
342 inline float regression_estimate_3()
const
344 return get(BSEC_OUTPUT_REGRESSION_ESTIMATE_3);
346 inline float regression_estimate_4()
const
348 return get(BSEC_OUTPUT_REGRESSION_ESTIMATE_4);
351 inline float raw_temperature()
const
353 return raw.temperature;
355 inline float raw_pressure()
const
359 inline float raw_humidity()
const
363 inline float raw_gas()
const
365 return raw.gas_resistance;
377class UnitBME688 :
public Component,
public PeriodicMeasurementAdapter<UnitBME688, bme688::Data> {
378 M5_UNIT_COMPONENT_HPP_BUILDER(
UnitBME688, 0x77);
390#if defined(UNIT_BME688_USING_BSEC2) || defined(DOXYGEN_PROCESS)
394 uint32_t
subscribe_bits{1U << BSEC_OUTPUT_IAQ | 1U << BSEC_OUTPUT_RAW_TEMPERATURE |
395 1U << BSEC_OUTPUT_RAW_PRESSURE | 1U << BSEC_OUTPUT_RAW_HUMIDITY |
396 1U << BSEC_OUTPUT_RAW_GAS | 1U << BSEC_OUTPUT_STABILIZATION_STATUS |
397 1U << BSEC_OUTPUT_RUN_IN_STATUS};
403 bme688::bsec2::SampleRate
sample_rate{bme688::bsec2::SampleRate::LowPower};
405#if !defined(UNIT_BME688_USING_BSEC2) || defined(DOXYGEN_PROCESS)
444 explicit UnitBME688(
const uint8_t addr = DEFAULT_ADDRESS);
449 virtual bool begin()
override;
450 virtual void update(
const bool force =
false)
override;
477 return _dev.amb_temp;
483#if defined(UNIT_BME688_USING_BSEC2)
488 inline float iaq()
const
490 return !empty() ? oldest().iaq() : std::numeric_limits<float>::quiet_NaN();
495 return !empty() ? oldest().temperature() : std::numeric_limits<float>::quiet_NaN();
500 return !empty() ? oldest().pressure() : std::numeric_limits<float>::quiet_NaN();
505 return !empty() ? oldest().humidity() : std::numeric_limits<float>::quiet_NaN();
508 inline float gas()
const
510 return !empty() ? oldest().gas() : std::numeric_limits<float>::quiet_NaN();
516 return !empty() ? oldest().raw_temperature() : std::numeric_limits<float>::quiet_NaN();
521 return !empty() ? oldest().raw_pressure() : std::numeric_limits<float>::quiet_NaN();
526 return !empty() ? oldest().raw_humidity() : std::numeric_limits<float>::quiet_NaN();
531 return !empty() ? oldest().raw_gas() : std::numeric_limits<float>::quiet_NaN();
542 inline uint8_t numberOfRawData()
const
553 inline const bme688::bme68xData* data(
const uint8_t idx)
555 return (idx < _num_of_data) ? &_raw_data[idx] :
nullptr;
562 _dev.amb_temp = temp;
710 return PeriodicMeasurementAdapter<UnitBME688, bme688::Data>::startPeriodicMeasurement(m);
712#if defined(UNIT_BME688_USING_BSEC2) || defined(DOXYGEN_PROCESS)
720 const bme688::bsec2::SampleRate sr = bme688::bsec2::SampleRate::LowPower)
722 return PeriodicMeasurementAdapter<UnitBME688, bme688::Data>::startPeriodicMeasurement(subscribe_bits, sr);
732 const bme688::bsec2::SampleRate sr = bme688::bsec2::SampleRate::LowPower)
744 return PeriodicMeasurementAdapter<UnitBME688, bme688::Data>::stopPeriodicMeasurement();
761#if defined(UNIT_BME688_USING_BSEC2) || defined(DOXYGEN_PROCESS)
770 return _temperatureOffset;
778 _temperatureOffset = offset;
787 return _bsec2_version;
797 bool bsec2SetConfig(
const uint8_t* cfg,
const size_t sz = BSEC_MAX_PROPERTY_BLOB_SIZE);
839 const bme688::bsec2::SampleRate sr)
850 return _bsec2_subscription & (1U << id);
859 return _bsec2_subscription;
883 static int8_t read_function(uint8_t reg_addr, uint8_t* reg_data, uint32_t length,
void* intf_ptr);
884 static int8_t write_function(uint8_t reg_addr,
const uint8_t* reg_data, uint32_t length,
void* intf_ptr);
887 bool stop_periodic_measurement();
888#if defined(UNIT_BME688_USING_BSEC2)
889 bool start_periodic_measurement(
const uint32_t subscribe_bits,
const bme688::bsec2::SampleRate sr);
892 bool write_mode_forced();
893 bool write_mode_parallel();
896 void update_bme688(
const bool force);
897 bool read_measurement();
898#if defined(UNIT_BME688_USING_BSEC2)
899 bool process_data(bsecOutputs& outouts,
const int64_t ns,
const bme688::bme68xData& data);
900 void update_bsec2(
const bool force);
903 inline virtual bool in_periodic()
const override
905 return _periodic || (_bsec2_subscription != 0);
914 bme688::bme68xData _raw_data[3]{};
915 uint8_t _num_of_data{};
916 bme688::bme68xDev _dev{};
917 bme688::bme68xConf _tphConf{};
918 bme688::bme68xHeatrConf _heaterConf{};
921 uint32_t _bsec2_subscription{};
923#if defined(UNIT_BME688_USING_BSEC2)
924 bsec_version_t _bsec2_version{};
925 std::unique_ptr<uint8_t> _bsec2_work{};
926 bsec_bme_settings_t _bsec2_settings{};
928 bme688::Mode _bsec2_mode{};
929 bme688::bsec2::SampleRate _bsec2_sr{};
931 bsecOutputs _outputs{};
932 float _temperatureOffset{};
935 std::unique_ptr<m5::container::CircularBuffer<bme688::Data>> _data{};
938 types::elapsed_time_t _can_measure_time{};
946constexpr uint8_t CHIP_ID{0xD0};
947constexpr uint8_t RESET{0xE0};
948constexpr uint8_t VARIANT_ID{0xF0};
950constexpr uint8_t IDAC_HEATER_0{0x50};
951constexpr uint8_t RES_HEAT_0{0x5A};
952constexpr uint8_t GAS_WAIT_0{0x64};
953constexpr uint8_t GAS_WAIT_SHARED{0x6E};
955constexpr uint8_t CTRL_GAS_0{0x70};
956constexpr uint8_t CTRL_GAS_1{0x71};
957constexpr uint8_t CTRL_HUMIDITY{0x72};
958constexpr uint8_t CTRL_MEASUREMENT{0x74};
959constexpr uint8_t CONFIG{0x75};
961constexpr uint8_t MEASUREMENT_STATUS_0{0x1D};
962constexpr uint8_t MEASUREMENT_STATUS_1{0x2E};
963constexpr uint8_t MEASUREMENT_STATUS_2{0x3F};
965constexpr uint8_t MEASUREMENT_GROUP_INDEX_0{0x1F};
966constexpr uint8_t MEASUREMENT_GROUP_INDEX_1{0x30};
967constexpr uint8_t MEASUREMENT_GROUP_INDEX_2{0x41};
969constexpr uint8_t UNIQUE_ID{0x83};
972constexpr uint8_t CALIBRATION_GROUP_0{0x8A};
973constexpr uint8_t CALIBRATION_GROUP_1{0xE1};
974constexpr uint8_t CALIBRATION_GROUP_2{0x00};
975constexpr uint8_t CALIBRATION_TEMPERATURE_1_LOW{0xE9};
976constexpr uint8_t CALIBRATION_TEMPERATURE_2_LOW{0x8A};
977constexpr uint8_t CALIBRATION_TEMPERATURE_3{0x8C};
978constexpr uint8_t CALIBRATION_PRESSURE_1_LOW{0x8E};
979constexpr uint8_t CALIBRATION_PRESSURE_2_LOW{0x90};
980constexpr uint8_t CALIBRATION_PRESSURE_3{0x92};
981constexpr uint8_t CALIBRATION_PRESSURE_4_LOW{0x94};
982constexpr uint8_t CALIBRATION_PRESSURE_5_LOW{0x96};
983constexpr uint8_t CALIBRATION_PRESSURE_6{0x99};
984constexpr uint8_t CALIBRATION_PRESSURE_7{0x98};
985constexpr uint8_t CALIBRATION_PRESSURE_8_LOW{0x9C};
986constexpr uint8_t CALIBRATION_PRESSURE_9_LOW{0x9E};
987constexpr uint8_t CALIBRATION_PRESSURE_10{0xA0};
988constexpr uint8_t CALIBRATION_HUMIDITY_12{0xE2};
989constexpr uint8_t CALIBRATION_HUMIDITY_1_HIGH{0xE3};
990constexpr uint8_t CALIBRATION_HUMIDITY_2_HIGH{0xE1};
991constexpr uint8_t CALIBRATION_HUMIDITY_3{0xE4};
992constexpr uint8_t CALIBRATION_HUMIDITY_4{0xE5};
993constexpr uint8_t CALIBRATION_HUMIDITY_5{0xE6};
994constexpr uint8_t CALIBRATION_HUMIDITY_6{0xE7};
995constexpr uint8_t CALIBRATION_HUMIDITY_7{0xE8};
996constexpr uint8_t CALIBRATION_GAS_1{0xED};
997constexpr uint8_t CALIBRATION_GAS_2_LOW{0xEB};
998constexpr uint8_t CALIBRATION_GAS_3{0xEE};
999constexpr uint8_t CALIBRATION_RES_HEAT_RANGE{0x02};
1000constexpr uint8_t CALIBRATION_RES_HEAT_VAL{0x00};
BME688 unit.
Definition unit_BME688.hpp:377
void config(const config_t &cfg)
Set the configration.
Definition unit_BME688.hpp:438
bool bsec2Unsubscribe(const bsec_virtual_sensor_t id)
Unsubscribe virtual sensor.
bool writeIIRFilter(const bme688::Filter f)
Write IIRFilter.
Definition unit_BME688.cpp:649
bool bsec2UpdateSubscription(const bsec_virtual_sensor_t *ss, const size_t len, const bme688::bsec2::SampleRate sr)
Subscribe to library virtual sensors outputs.
Definition unit_BME688.hpp:838
bool readHeaterSetting(bme688::bme68xHeatrConf &hs)
Read heater setting.
Definition unit_BME688.cpp:662
bool bsec2SetState(const uint8_t *state)
Restore the internal state.
bool readOversamplingPressure(bme688::Oversampling &os)
Read pressure oversampling.
Definition unit_BME688.cpp:560
const bsec_version_t & bsec2Version() const
Gets the BSEC2 library version.
Definition unit_BME688.hpp:785
void setAambientTemperature(const int8_t temp)
Sets the ambient temperature.
Definition unit_BME688.hpp:560
bool readIIRFilter(bme688::Filter &f)
Read IIRFilter.
Definition unit_BME688.cpp:582
bool writeCalibration(const bme688::bme68xCalibration &c)
write calibration
Definition unit_BME688.cpp:483
bool readUniqueID(uint32_t &id)
Read unique ID.
Definition unit_BME688.cpp:410
float bsec2GetTemperatureOffset() const
Gets the temperature offset(Celsius)
Definition unit_BME688.hpp:768
bool measureSingleShot(bme688::bme68xData &data)
Take a single measurement.
Definition unit_BME688.cpp:696
bool bsec2GetState(uint8_t *state, uint32_t &actualSize)
Retrieve the current internal library state.
float gas() const
Oldest measured gas (Ohm)
Definition unit_BME688.hpp:529
bool softReset()
Software reset.
Definition unit_BME688.cpp:424
bool startPeriodicMeasurement(const bsec_virtual_sensor_t *ss, const size_t len, const bme688::bsec2::SampleRate sr=bme688::bsec2::SampleRate::LowPower)
Start periodic measurement using BSEC2.
Definition unit_BME688.hpp:731
bool bsec2IsSubscribed(const bsec_virtual_sensor_t id)
is virtual sensor Subscribed?
Definition unit_BME688.hpp:848
const bme688::bme68xConf & tphSetting() const
Gets the TPH setting.
Definition unit_BME688.hpp:465
bool readMode(bme688::Mode &m)
Read operation mode.
Definition unit_BME688.cpp:685
bool bsec2UpdateSubscription(const uint32_t sensorBits, const bme688::bsec2::SampleRate sr)
Subscribe to library virtual sensors outputs.
bool selfTest()
Self-test.
Definition unit_BME688.cpp:431
bme688::Mode mode() const
Current mode.
Definition unit_BME688.hpp:455
bool readOversamplingHumidity(bme688::Oversampling &os)
Read humidity oversampling.
Definition unit_BME688.cpp:571
bool readOversamplingTemperature(bme688::Oversampling &os)
Read temperature oversampling.
Definition unit_BME688.cpp:549
bool writeOversamplingTemperature(const bme688::Oversampling os)
Write temperature oversampling.
Definition unit_BME688.cpp:610
bool writeTPHSetting(const bme688::bme68xConf &s)
Write TPH setting.
Definition unit_BME688.cpp:539
bool bsec2GetConfig(uint8_t *cfg, uint32_t &actualSize)
Retrieve the current library configuration.
bool writeMode(const bme688::Mode m)
Write operation mode.
Definition unit_BME688.cpp:676
void bsec2SetTemperatureOffset(const float offset)
Set the temperature offset(Celsius)
Definition unit_BME688.hpp:776
bool bsec2SetConfig(const uint8_t *cfg, const size_t sz=BSEC_MAX_PROPERTY_BLOB_SIZE)
Update algorithm configuration parameters Update bsec2 configuration settings.
bool writeOversamplingHumidity(const bme688::Oversampling os)
Write humidity oversampling.
Definition unit_BME688.cpp:636
bool writeOversamplingPressure(const bme688::Oversampling os)
Write pressure oversampling.
Definition unit_BME688.cpp:623
bool bsec2Subscribe(const bsec_virtual_sensor_t id)
Subscribe virtual sensor.
bool startPeriodicMeasurement(const bme688::Mode m)
Start periodic measurement without BSEC2.
Definition unit_BME688.hpp:708
bool writeHeaterSetting(const bme688::Mode mode, const bme688::bme68xHeatrConf &hs)
Write heater setting.
Definition unit_BME688.cpp:667
bool startPeriodicMeasurement(const uint32_t subscribe_bits, const bme688::bsec2::SampleRate sr=bme688::bsec2::SampleRate::LowPower)
Start periodic measurement using BSEC2.
Definition unit_BME688.hpp:719
const bme688::bme68xCalibration & calibration() const
Gets the Calibration.
Definition unit_BME688.hpp:460
bool readCalibration(bme688::bme68xCalibration &c)
Read calibration.
Definition unit_BME688.cpp:436
bool bsec2UnsubscribeAll()
Unsubacribe currentt all sensors.
uint32_t bsec2Subscription() const
Gets the subscription bits.
Definition unit_BME688.hpp:857
float pressure() const
Oldest measured pressure (Pa)
Definition unit_BME688.hpp:519
const bme688::bme68xHeatrConf & heaterSetting() const
Gets the heater setiing.
Definition unit_BME688.hpp:470
bool stopPeriodicMeasurement()
Stop periodic measurement.
Definition unit_BME688.hpp:742
int8_t ambientTemperature() const
Gets the ambient temperature.
Definition unit_BME688.hpp:475
uint32_t calculateMeasurementInterval(const bme688::Mode mode, const bme688::bme68xConf &s)
Calculation of measurement intervals without heater.
Definition unit_BME688.cpp:691
bool writeOversampling(const bme688::Oversampling t, const bme688::Oversampling p, const bme688::Oversampling h)
Wite oversamplings.
Definition unit_BME688.cpp:593
config_t config()
Gets the configration.
Definition unit_BME688.hpp:433
float humidity() const
Oldest measured humidity (%)
Definition unit_BME688.hpp:524
float temperature() const
Oldest measured temperature (Celsius)
Definition unit_BME688.hpp:514
bool readTPHSetting(bme688::bme68xConf &s)
Read TPH setting.
Definition unit_BME688.cpp:534
Top level namespace of M5stack.
Settings for begin.
Definition unit_BME688.hpp:385
uint32_t subscribe_bits
Subscribe BSEC2 sensors bits if start on begin.
Definition unit_BME688.hpp:394
bme688::ODR odr
Standby time between sequential mode measurement profiles if start on begin.
Definition unit_BME688.hpp:419
bme688::Oversampling oversampling_pressure
Pressure oversampling if start on begin.
Definition unit_BME688.hpp:413
bme688::Oversampling oversampling_temperature
Temperature oversampling if start on begin.
Definition unit_BME688.hpp:411
uint16_t heater_temperature
The heater temperature for forced mode degree Celsius if start on begin.
Definition unit_BME688.hpp:423
bme688::Mode mode
Measurement mode if start on begin.
Definition unit_BME688.hpp:409
bool start_periodic
Start periodic measurement on begin?
Definition unit_BME688.hpp:387
int8_t ambient_temperature
ambient temperature
Definition unit_BME688.hpp:389
bme688::Filter filter
Filter coefficient if start on begin.
Definition unit_BME688.hpp:417
bool heater_enable
Enable gas measurement if start on begin.
Definition unit_BME688.hpp:421
bme688::bsec2::SampleRate sample_rate
Sampling rate for BSEC2 if start on begin.
Definition unit_BME688.hpp:403
bme688::Oversampling oversampling_humidity
Humidity oversampling if start on begin.
Definition unit_BME688.hpp:415
uint16_t heater_duration
The heating duration for forced mode in milliseconds if start on begin.
Definition unit_BME688.hpp:425
Measurement data group.
Definition unit_BME688.hpp:256
GasSensor heater-on time.
Definition unit_BME688.hpp:141
uint8_t value
Use the value as it is in parallel mode.
Definition unit_BME688.hpp:194
static uint16_t to(const uint8_t v)
Conversion from register value to duration for Force/Sequencial mode.
Definition unit_BME688.hpp:188
Factor
Multiplier in Forced mode.
Definition unit_BME688.hpp:149
static uint8_t from(const uint16_t duration)
Conversion from duration to register value for Force/Sequencial mode.
Definition unit_BME688.hpp:176
Setting for gas heater.
Definition unit_BME688.hpp:76
struct bme68x_data bme68xData
Raw data.
Definition unit_BME688.hpp:60
struct bme68x_conf bme68xConf
Setting for temperature, pressure, humidity...
Definition unit_BME688.hpp:70
Mode
Operation mode same as BME68X_xxx_MODE.
Definition unit_BME688.hpp:46
@ Sleep
No measurements are performed.
@ Parallel
Multiple TPHG cycles are performed.
@ Forced
Single TPHG cycle is performed.
struct bme68x_calib_data bme68xCalibration
Calibration parameters.
Definition unit_BME688.hpp:89
Filter
IIR Filter setting.
Definition unit_BME688.hpp:109
ODR
bme68xConf::odr settings (standbytime Unit:ms)
Definition unit_BME688.hpp:124
Oversampling
Definition unit_BME688.hpp:96
struct bme68x_dev bme68xDev
bme68x device
Definition unit_BME688.hpp:65