BME680.c 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431
  1. /*
  2. Unitemp - Universal temperature reader
  3. Copyright (C) 2022-2023 Victor Nikitchuk (https://github.com/quen0n)
  4. Contributed by g0gg0 (https://github.com/g3gg0)
  5. This program is free software: you can redistribute it and/or modify
  6. it under the terms of the GNU General Public License as published by
  7. the Free Software Foundation, either version 3 of the License, or
  8. (at your option) any later version.
  9. This program is distributed in the hope that it will be useful,
  10. but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. GNU General Public License for more details.
  13. You should have received a copy of the GNU General Public License
  14. along with this program. If not, see <https://www.gnu.org/licenses/>.
  15. */
  16. #include "BME680.h"
  17. const SensorType BME680 = {
  18. .typename = "BME680",
  19. .interface = &I2C,
  20. .datatype = UT_TEMPERATURE | UT_HUMIDITY | UT_PRESSURE,
  21. .pollingInterval = 500,
  22. .allocator = unitemp_BME680_alloc,
  23. .mem_releaser = unitemp_BME680_free,
  24. .initializer = unitemp_BME680_init,
  25. .deinitializer = unitemp_BME680_deinit,
  26. .updater = unitemp_BME680_update};
  27. //Интервал обновления калибровочных значений
  28. #define BOSCH_CAL_UPDATE_INTERVAL 60000
  29. #define BME680_ID 0x61
  30. #define BME680_I2C_ADDR_MIN (0x76 << 1)
  31. #define BME680_I2C_ADDR_MAX (0x77 << 1)
  32. #define BME680_REG_STATUS 0x1D
  33. #define BME680_REG_CTRL_MEAS 0x74
  34. #define BME680_REG_CONFIG 0x75
  35. #define BME680_REG_CTRL_HUM 0x72
  36. //Преддескретизация температуры
  37. #define BME680_TEMP_OVERSAMPLING_SKIP 0b00000000
  38. #define BME680_TEMP_OVERSAMPLING_1 0b00100000
  39. #define BME680_TEMP_OVERSAMPLING_2 0b01000000
  40. #define BME680_TEMP_OVERSAMPLING_4 0b01100000
  41. #define BME680_TEMP_OVERSAMPLING_8 0b10000000
  42. #define BME680_TEMP_OVERSAMPLING_16 0b10100000
  43. //Преддескретизация давления
  44. #define BME680_PRESS_OVERSAMPLING_SKIP 0b00000000
  45. #define BME680_PRESS_OVERSAMPLING_1 0b00000100
  46. #define BME680_PRESS_OVERSAMPLING_2 0b00001000
  47. #define BME680_PRESS_OVERSAMPLING_4 0b00001100
  48. #define BME680_PRESS_OVERSAMPLING_8 0b00010000
  49. #define BME680_PRESS_OVERSAMPLING_16 0b00010100
  50. //Преддескретизация влажности
  51. #define BME680_HUM_OVERSAMPLING_SKIP 0b00000000
  52. #define BME680_HUM_OVERSAMPLING_1 0b00000001
  53. #define BME680_HUM_OVERSAMPLING_2 0b00000010
  54. #define BME680_HUM_OVERSAMPLING_4 0b00000011
  55. #define BME680_HUM_OVERSAMPLING_8 0b00000100
  56. #define BME680_HUM_OVERSAMPLING_16 0b00000101
  57. //Режимы работы датчика
  58. #define BME680_MODE_SLEEP 0b00000000 //Наелся и спит
  59. #define BME680_MODE_FORCED 0b00000001 //Обновляет значения 1 раз, после чего уходит в сон
  60. //Коэффициент фильтрации значений
  61. #define BME680_FILTER_COEFF_1 0b00000000
  62. #define BME680_FILTER_COEFF_2 0b00000100
  63. #define BME680_FILTER_COEFF_4 0b00001000
  64. #define BME680_FILTER_COEFF_8 0b00001100
  65. #define BME680_FILTER_COEFF_16 0b00010000
  66. //Разрешить работу по SPI
  67. #define BME680_SPI_3W_ENABLE 0b00000001
  68. #define BME680_SPI_3W_DISABLE 0b00000000
  69. /* https://github.com/boschsensortec/BME680_driver/blob/master/bme680.c or
  70. https://github.com/boschsensortec/BME68x-Sensor-API */
  71. static float BME680_compensate_temperature(I2CSensor* i2c_sensor, int32_t temp_adc) {
  72. BME680_instance* bme680_instance = (BME680_instance*)i2c_sensor->sensorInstance;
  73. float var1 = 0;
  74. float var2 = 0;
  75. float calc_temp = 0;
  76. /* calculate var1 data */
  77. var1 =
  78. ((((float)temp_adc / 16384.0f) - ((float)bme680_instance->temp_cal.dig_T1 / 1024.0f)) *
  79. ((float)bme680_instance->temp_cal.dig_T2));
  80. /* calculate var2 data */
  81. var2 =
  82. (((((float)temp_adc / 131072.0f) - ((float)bme680_instance->temp_cal.dig_T1 / 8192.0f)) *
  83. (((float)temp_adc / 131072.0f) - ((float)bme680_instance->temp_cal.dig_T1 / 8192.0f))) *
  84. ((float)bme680_instance->temp_cal.dig_T3 * 16.0f));
  85. /* t_fine value*/
  86. bme680_instance->t_fine = (var1 + var2);
  87. /* compensated temperature data*/
  88. calc_temp = ((bme680_instance->t_fine) / 5120.0f);
  89. return calc_temp;
  90. }
  91. static float BME680_compensate_pressure(I2CSensor* i2c_sensor, int32_t pres_adc) {
  92. BME680_instance* bme680_instance = (BME680_instance*)i2c_sensor->sensorInstance;
  93. float var1;
  94. float var2;
  95. float var3;
  96. float calc_pres;
  97. var1 = (((float)bme680_instance->t_fine / 2.0f) - 64000.0f);
  98. var2 = var1 * var1 * (((float)bme680_instance->press_cal.dig_P6) / (131072.0f));
  99. var2 = var2 + (var1 * ((float)bme680_instance->press_cal.dig_P5) * 2.0f);
  100. var2 = (var2 / 4.0f) + (((float)bme680_instance->press_cal.dig_P4) * 65536.0f);
  101. var1 =
  102. (((((float)bme680_instance->press_cal.dig_P3 * var1 * var1) / 16384.0f) +
  103. ((float)bme680_instance->press_cal.dig_P2 * var1)) /
  104. 524288.0f);
  105. var1 = ((1.0f + (var1 / 32768.0f)) * ((float)bme680_instance->press_cal.dig_P1));
  106. calc_pres = (1048576.0f - ((float)pres_adc));
  107. /* Avoid exception caused by division by zero */
  108. if((int)var1 != 0) {
  109. calc_pres = (((calc_pres - (var2 / 4096.0f)) * 6250.0f) / var1);
  110. var1 =
  111. (((float)bme680_instance->press_cal.dig_P9) * calc_pres * calc_pres) / 2147483648.0f;
  112. var2 = calc_pres * (((float)bme680_instance->press_cal.dig_P8) / 32768.0f);
  113. var3 =
  114. ((calc_pres / 256.0f) * (calc_pres / 256.0f) * (calc_pres / 256.0f) *
  115. (bme680_instance->press_cal.dig_P10 / 131072.0f));
  116. calc_pres =
  117. (calc_pres +
  118. (var1 + var2 + var3 + ((float)bme680_instance->press_cal.dig_P7 * 128.0f)) / 16.0f);
  119. } else {
  120. calc_pres = 0;
  121. }
  122. return calc_pres;
  123. }
  124. static float BME680_compensate_humidity(I2CSensor* i2c_sensor, int32_t hum_adc) {
  125. BME680_instance* bme680_instance = (BME680_instance*)i2c_sensor->sensorInstance;
  126. float calc_hum;
  127. float var1;
  128. float var2;
  129. float var3;
  130. float var4;
  131. float temp_comp;
  132. /* compensated temperature data*/
  133. temp_comp = ((bme680_instance->t_fine) / 5120.0f);
  134. var1 =
  135. (float)((float)hum_adc) - (((float)bme680_instance->hum_cal.dig_H1 * 16.0f) +
  136. (((float)bme680_instance->hum_cal.dig_H3 / 2.0f) * temp_comp));
  137. var2 = var1 *
  138. ((float)(((float)bme680_instance->hum_cal.dig_H2 / 262144.0f) *
  139. (1.0f + (((float)bme680_instance->hum_cal.dig_H4 / 16384.0f) * temp_comp) +
  140. (((float)bme680_instance->hum_cal.dig_H5 / 1048576.0f) * temp_comp * temp_comp))));
  141. var3 = (float)bme680_instance->hum_cal.dig_H6 / 16384.0f;
  142. var4 = (float)bme680_instance->hum_cal.dig_H7 / 2097152.0f;
  143. calc_hum = var2 + ((var3 + (var4 * temp_comp)) * var2 * var2);
  144. if(calc_hum > 100.0f) {
  145. calc_hum = 100.0f;
  146. } else if(calc_hum < 0.0f) {
  147. calc_hum = 0.0f;
  148. }
  149. return calc_hum;
  150. }
  151. /* https://github.com/boschsensortec/BME680_driver/blob/master/bme680_defs.h */
  152. #define BME680_COEFF_SIZE UINT8_C(41)
  153. #define BME680_COEFF_ADDR1_LEN UINT8_C(25)
  154. #define BME680_COEFF_ADDR2_LEN UINT8_C(16)
  155. #define BME680_COEFF_ADDR1 UINT8_C(0x89)
  156. #define BME680_COEFF_ADDR2 UINT8_C(0xe1)
  157. #define BME680_CONCAT_BYTES(msb, lsb) (((uint16_t)msb << 8) | (uint16_t)lsb)
  158. #define BME680_T2_LSB_REG (1)
  159. #define BME680_T2_MSB_REG (2)
  160. #define BME680_T3_REG (3)
  161. #define BME680_P1_LSB_REG (5)
  162. #define BME680_P1_MSB_REG (6)
  163. #define BME680_P2_LSB_REG (7)
  164. #define BME680_P2_MSB_REG (8)
  165. #define BME680_P3_REG (9)
  166. #define BME680_P4_LSB_REG (11)
  167. #define BME680_P4_MSB_REG (12)
  168. #define BME680_P5_LSB_REG (13)
  169. #define BME680_P5_MSB_REG (14)
  170. #define BME680_P7_REG (15)
  171. #define BME680_P6_REG (16)
  172. #define BME680_P8_LSB_REG (19)
  173. #define BME680_P8_MSB_REG (20)
  174. #define BME680_P9_LSB_REG (21)
  175. #define BME680_P9_MSB_REG (22)
  176. #define BME680_P10_REG (23)
  177. #define BME680_H2_MSB_REG (25)
  178. #define BME680_H2_LSB_REG (26)
  179. #define BME680_H1_LSB_REG (26)
  180. #define BME680_H1_MSB_REG (27)
  181. #define BME680_H3_REG (28)
  182. #define BME680_H4_REG (29)
  183. #define BME680_H5_REG (30)
  184. #define BME680_H6_REG (31)
  185. #define BME680_H7_REG (32)
  186. #define BME680_T1_LSB_REG (33)
  187. #define BME680_T1_MSB_REG (34)
  188. #define BME680_GH2_LSB_REG (35)
  189. #define BME680_GH2_MSB_REG (36)
  190. #define BME680_GH1_REG (37)
  191. #define BME680_GH3_REG (38)
  192. #define BME680_HUM_REG_SHIFT_VAL UINT8_C(4)
  193. #define BME680_BIT_H1_DATA_MSK UINT8_C(0x0F)
  194. static bool BME680_readCalValues(I2CSensor* i2c_sensor) {
  195. BME680_instance* bme680_instance = (BME680_instance*)i2c_sensor->sensorInstance;
  196. uint8_t coeff_array[BME680_COEFF_SIZE] = {0};
  197. if(!unitemp_i2c_readRegArray(
  198. i2c_sensor, BME680_COEFF_ADDR1, BME680_COEFF_ADDR1_LEN, &coeff_array[0]))
  199. return false;
  200. if(!unitemp_i2c_readRegArray(
  201. i2c_sensor,
  202. BME680_COEFF_ADDR2,
  203. BME680_COEFF_ADDR2_LEN,
  204. &coeff_array[BME680_COEFF_ADDR1_LEN]))
  205. return false;
  206. /* Temperature related coefficients */
  207. bme680_instance->temp_cal.dig_T1 = (uint16_t)(BME680_CONCAT_BYTES(
  208. coeff_array[BME680_T1_MSB_REG], coeff_array[BME680_T1_LSB_REG]));
  209. bme680_instance->temp_cal.dig_T2 = (int16_t)(BME680_CONCAT_BYTES(
  210. coeff_array[BME680_T2_MSB_REG], coeff_array[BME680_T2_LSB_REG]));
  211. bme680_instance->temp_cal.dig_T3 = (int8_t)(coeff_array[BME680_T3_REG]);
  212. /* Pressure related coefficients */
  213. bme680_instance->press_cal.dig_P1 = (uint16_t)(BME680_CONCAT_BYTES(
  214. coeff_array[BME680_P1_MSB_REG], coeff_array[BME680_P1_LSB_REG]));
  215. bme680_instance->press_cal.dig_P2 = (int16_t)(BME680_CONCAT_BYTES(
  216. coeff_array[BME680_P2_MSB_REG], coeff_array[BME680_P2_LSB_REG]));
  217. bme680_instance->press_cal.dig_P3 = (int8_t)coeff_array[BME680_P3_REG];
  218. bme680_instance->press_cal.dig_P4 = (int16_t)(BME680_CONCAT_BYTES(
  219. coeff_array[BME680_P4_MSB_REG], coeff_array[BME680_P4_LSB_REG]));
  220. bme680_instance->press_cal.dig_P5 = (int16_t)(BME680_CONCAT_BYTES(
  221. coeff_array[BME680_P5_MSB_REG], coeff_array[BME680_P5_LSB_REG]));
  222. bme680_instance->press_cal.dig_P6 = (int8_t)(coeff_array[BME680_P6_REG]);
  223. bme680_instance->press_cal.dig_P7 = (int8_t)(coeff_array[BME680_P7_REG]);
  224. bme680_instance->press_cal.dig_P8 = (int16_t)(BME680_CONCAT_BYTES(
  225. coeff_array[BME680_P8_MSB_REG], coeff_array[BME680_P8_LSB_REG]));
  226. bme680_instance->press_cal.dig_P9 = (int16_t)(BME680_CONCAT_BYTES(
  227. coeff_array[BME680_P9_MSB_REG], coeff_array[BME680_P9_LSB_REG]));
  228. bme680_instance->press_cal.dig_P10 = (uint8_t)(coeff_array[BME680_P10_REG]);
  229. /* Humidity related coefficients */
  230. bme680_instance->hum_cal.dig_H1 =
  231. (uint16_t)(((uint16_t)coeff_array[BME680_H1_MSB_REG] << BME680_HUM_REG_SHIFT_VAL) | (coeff_array[BME680_H1_LSB_REG] & BME680_BIT_H1_DATA_MSK));
  232. bme680_instance->hum_cal.dig_H2 =
  233. (uint16_t)(((uint16_t)coeff_array[BME680_H2_MSB_REG] << BME680_HUM_REG_SHIFT_VAL) | ((coeff_array[BME680_H2_LSB_REG]) >> BME680_HUM_REG_SHIFT_VAL));
  234. bme680_instance->hum_cal.dig_H3 = (int8_t)coeff_array[BME680_H3_REG];
  235. bme680_instance->hum_cal.dig_H4 = (int8_t)coeff_array[BME680_H4_REG];
  236. bme680_instance->hum_cal.dig_H5 = (int8_t)coeff_array[BME680_H5_REG];
  237. bme680_instance->hum_cal.dig_H6 = (uint8_t)coeff_array[BME680_H6_REG];
  238. bme680_instance->hum_cal.dig_H7 = (int8_t)coeff_array[BME680_H7_REG];
  239. /* Gas heater related coefficients */
  240. bme680_instance->gas_cal.dig_GH1 = (int8_t)coeff_array[BME680_GH1_REG];
  241. bme680_instance->gas_cal.dig_GH2 = (int16_t)(BME680_CONCAT_BYTES(
  242. coeff_array[BME680_GH2_MSB_REG], coeff_array[BME680_GH2_LSB_REG]));
  243. bme680_instance->gas_cal.dig_GH3 = (int8_t)coeff_array[BME680_GH3_REG];
  244. #ifdef UNITEMP_DEBUG
  245. FURI_LOG_D(
  246. APP_NAME,
  247. "Sensor BME680 T1-T3: %d, %d, %d",
  248. bme680_instance->temp_cal.dig_T1,
  249. bme680_instance->temp_cal.dig_T2,
  250. bme680_instance->temp_cal.dig_T3);
  251. FURI_LOG_D(
  252. APP_NAME,
  253. "Sensor BME680: P1-P10: %d, %d, %d, %d, %d, %d, %d, %d, %d, %d",
  254. bme680_instance->press_cal.dig_P1,
  255. bme680_instance->press_cal.dig_P2,
  256. bme680_instance->press_cal.dig_P3,
  257. bme680_instance->press_cal.dig_P4,
  258. bme680_instance->press_cal.dig_P5,
  259. bme680_instance->press_cal.dig_P6,
  260. bme680_instance->press_cal.dig_P7,
  261. bme680_instance->press_cal.dig_P8,
  262. bme680_instance->press_cal.dig_P9,
  263. bme680_instance->press_cal.dig_P10);
  264. FURI_LOG_D(
  265. APP_NAME,
  266. "Sensor BME680: H1-H7: %d, %d, %d, %d, %d, %d, %d",
  267. bme680_instance->hum_cal.dig_H1,
  268. bme680_instance->hum_cal.dig_H2,
  269. bme680_instance->hum_cal.dig_H3,
  270. bme680_instance->hum_cal.dig_H4,
  271. bme680_instance->hum_cal.dig_H5,
  272. bme680_instance->hum_cal.dig_H6,
  273. bme680_instance->hum_cal.dig_H7);
  274. FURI_LOG_D(
  275. APP_NAME,
  276. "Sensor BME680 GH1-GH3: %d, %d, %d",
  277. bme680_instance->gas_cal.dig_GH1,
  278. bme680_instance->gas_cal.dig_GH2,
  279. bme680_instance->gas_cal.dig_GH3);
  280. #endif
  281. bme680_instance->last_cal_update_time = furi_get_tick();
  282. return true;
  283. }
  284. static bool BME680_isMeasuring(Sensor* sensor) {
  285. I2CSensor* i2c_sensor = (I2CSensor*)sensor->instance;
  286. return (bool)(unitemp_i2c_readReg(i2c_sensor, BME680_REG_STATUS) & 0x20);
  287. }
  288. bool unitemp_BME680_alloc(Sensor* sensor, char* args) {
  289. UNUSED(args);
  290. I2CSensor* i2c_sensor = (I2CSensor*)sensor->instance;
  291. BME680_instance* bme680_instance = malloc(sizeof(BME680_instance));
  292. if(bme680_instance == NULL) {
  293. FURI_LOG_E(APP_NAME, "Failed to allocation sensor %s instance", sensor->name);
  294. return false;
  295. }
  296. if(sensor->type == &BME680) bme680_instance->chip_id = BME680_ID;
  297. i2c_sensor->sensorInstance = bme680_instance;
  298. i2c_sensor->minI2CAdr = BME680_I2C_ADDR_MIN;
  299. i2c_sensor->maxI2CAdr = BME680_I2C_ADDR_MAX;
  300. return true;
  301. }
  302. bool unitemp_BME680_init(Sensor* sensor) {
  303. I2CSensor* i2c_sensor = (I2CSensor*)sensor->instance;
  304. //Перезагрузка
  305. unitemp_i2c_writeReg(i2c_sensor, 0xE0, 0xB6);
  306. //Чтение ID датчика
  307. uint8_t id = unitemp_i2c_readReg(i2c_sensor, 0xD0);
  308. if(id != BME680_ID) {
  309. FURI_LOG_E(
  310. APP_NAME,
  311. "Sensor %s returned wrong ID 0x%02X, expected 0x%02X",
  312. sensor->name,
  313. id,
  314. BME680_ID);
  315. return false;
  316. }
  317. unitemp_i2c_writeReg(
  318. i2c_sensor,
  319. BME680_REG_CTRL_HUM,
  320. (unitemp_i2c_readReg(i2c_sensor, BME680_REG_CTRL_HUM) & ~7) | BME680_HUM_OVERSAMPLING_1);
  321. unitemp_i2c_writeReg(
  322. i2c_sensor,
  323. BME680_REG_CTRL_MEAS,
  324. BME680_TEMP_OVERSAMPLING_2 | BME680_PRESS_OVERSAMPLING_4 | BME680_MODE_FORCED);
  325. //Настройка периода опроса и фильтрации значений
  326. unitemp_i2c_writeReg(
  327. i2c_sensor, BME680_REG_CONFIG, BME680_FILTER_COEFF_16 | BME680_SPI_3W_DISABLE);
  328. //Чтение калибровочных значений
  329. if(!BME680_readCalValues(i2c_sensor)) {
  330. FURI_LOG_E(APP_NAME, "Failed to read calibration values sensor %s", sensor->name);
  331. return false;
  332. }
  333. return true;
  334. }
  335. bool unitemp_BME680_deinit(Sensor* sensor) {
  336. I2CSensor* i2c_sensor = (I2CSensor*)sensor->instance;
  337. //Перевод в сон
  338. unitemp_i2c_writeReg(i2c_sensor, BME680_REG_CTRL_MEAS, BME680_MODE_SLEEP);
  339. return true;
  340. }
  341. UnitempStatus unitemp_BME680_update(Sensor* sensor) {
  342. I2CSensor* i2c_sensor = (I2CSensor*)sensor->instance;
  343. BME680_instance* instance = i2c_sensor->sensorInstance;
  344. uint32_t t = furi_get_tick();
  345. uint8_t buff[3];
  346. //Проверка инициализированности датчика
  347. unitemp_i2c_readRegArray(i2c_sensor, 0xF4, 2, buff);
  348. if(buff[0] == 0) {
  349. FURI_LOG_W(APP_NAME, "Sensor %s is not initialized!", sensor->name);
  350. return UT_SENSORSTATUS_ERROR;
  351. }
  352. unitemp_i2c_writeReg(
  353. i2c_sensor,
  354. BME680_REG_CTRL_MEAS,
  355. unitemp_i2c_readReg(i2c_sensor, BME680_REG_CTRL_MEAS) | 1);
  356. while(BME680_isMeasuring(sensor)) {
  357. if(furi_get_tick() - t > 100) {
  358. return UT_SENSORSTATUS_TIMEOUT;
  359. }
  360. }
  361. if(furi_get_tick() - instance->last_cal_update_time > BOSCH_CAL_UPDATE_INTERVAL) {
  362. BME680_readCalValues(i2c_sensor);
  363. }
  364. if(!unitemp_i2c_readRegArray(i2c_sensor, 0x1F, 3, buff)) return UT_SENSORSTATUS_TIMEOUT;
  365. int32_t adc_P = ((int32_t)buff[0] << 12) | ((int32_t)buff[1] << 4) | ((int32_t)buff[2] >> 4);
  366. if(!unitemp_i2c_readRegArray(i2c_sensor, 0x22, 3, buff)) return UT_SENSORSTATUS_TIMEOUT;
  367. int32_t adc_T = ((int32_t)buff[0] << 12) | ((int32_t)buff[1] << 4) | ((int32_t)buff[2] >> 4);
  368. if(!unitemp_i2c_readRegArray(i2c_sensor, 0x25, 2, buff)) return UT_SENSORSTATUS_TIMEOUT;
  369. int32_t adc_H = ((uint16_t)buff[0] << 8) | buff[1];
  370. sensor->temp = BME680_compensate_temperature(i2c_sensor, adc_T);
  371. sensor->pressure = BME680_compensate_pressure(i2c_sensor, adc_P);
  372. sensor->hum = BME680_compensate_humidity(i2c_sensor, adc_H);
  373. return UT_SENSORSTATUS_OK;
  374. }
  375. bool unitemp_BME680_free(Sensor* sensor) {
  376. I2CSensor* i2c_sensor = (I2CSensor*)sensor->instance;
  377. free(i2c_sensor->sensorInstance);
  378. return true;
  379. }