sht30.c 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250
  1. #include <sys/cdefs.h>
  2. /**
  3. * @Author: 李建
  4. * @Date: 2025/4/22 13:48
  5. * Description: 温湿度传感器
  6. * Copyright: Copyright (©) 2025 永续绿建. All rights reserved.
  7. */
  8. #include <esp_log.h>
  9. #include <esp_wifi.h>
  10. #include "sht30.h"
  11. #include "driver/i2c.h"
  12. #include "main.h"
  13. #include "lvgl_port.h"
  14. #include "wifi.h"
  15. #include "lcd_st7701.h"
  16. static const char *TAG = "SHT30";
  17. #define DRIFT_RATE 0.1524 // 根据历史数据计算的漂移率 (°C/min)
  18. #define ACK_CHECK_EN 0x1 /*!< I2C master will check ack from slave*/
  19. #define ACK_CHECK_DIS 0x0 /*!< I2C master will not check ack from slave */
  20. #define ACK_VAL 0x0 /*!< I2C ack value */
  21. #define NACK_VAL 0x1 /*!< I2C nack value */
  22. int current_seconds;
  23. sht30_data_t sht30Data = {0};
  24. // 温度修正函数
  25. static double correct_temperature(double current_temp) {
  26. static const int start_seconds = 0;
  27. // 计算经过时间(秒)
  28. double elapsed_seconds = (double) (current_seconds - start_seconds);
  29. // 计算修正量(漂移率 × 经过分钟数)
  30. double correction = DRIFT_RATE * (elapsed_seconds / 60.0);
  31. // 应用修正
  32. return current_temp - correction;
  33. }
  34. static esp_err_t i2c_master_init(void) {
  35. int i2c_master_port = I2C_MASTER_NUM;
  36. i2c_config_t conf = {
  37. .mode = I2C_MODE_MASTER,
  38. .sda_io_num = I2C_MASTER_SDA_IO,
  39. .scl_io_num = I2C_MASTER_SCL_IO,
  40. .sda_pullup_en = GPIO_PULLUP_ENABLE,
  41. .scl_pullup_en = GPIO_PULLUP_ENABLE,
  42. .master.clk_speed = I2C_MASTER_FREQ_HZ,
  43. };
  44. i2c_param_config(i2c_master_port, &conf);
  45. return i2c_driver_install(i2c_master_port, conf.mode, I2C_MASTER_RX_BUF_DISABLE, I2C_MASTER_TX_BUF_DISABLE, 0);
  46. }
  47. esp_err_t
  48. i2c_master_write_slave_reg(i2c_port_t i2c_num, uint8_t slave_addr, uint8_t reg_addr, uint8_t *data_wr, size_t size,
  49. TickType_t ticks_to_wait) {
  50. i2c_cmd_handle_t cmd = i2c_cmd_link_create();
  51. i2c_master_start(cmd);
  52. i2c_master_write_byte(cmd, (slave_addr << 1) | I2C_MASTER_WRITE, ACK_CHECK_EN);
  53. i2c_master_write_byte(cmd, reg_addr, ACK_CHECK_EN);
  54. i2c_master_write(cmd, data_wr, size, ACK_CHECK_EN);
  55. i2c_master_stop(cmd);
  56. esp_err_t ret = i2c_master_cmd_begin(i2c_num, cmd, ticks_to_wait);
  57. i2c_cmd_link_delete(cmd);
  58. return ret;
  59. }
  60. esp_err_t i2c_master_read_slave_reg_16bit(i2c_port_t i2c_num, uint8_t slave_addr, uint16_t reg_addr, uint8_t *data_rd,
  61. size_t size, TickType_t ticks_to_wait) {
  62. if (size == 0) {
  63. return ESP_OK;
  64. }
  65. i2c_cmd_handle_t cmd = i2c_cmd_link_create();
  66. i2c_master_start(cmd);
  67. i2c_master_write_byte(cmd, (slave_addr << 1) | I2C_MASTER_WRITE, ACK_CHECK_EN);
  68. i2c_master_write_byte(cmd, reg_addr >> 8, ACK_CHECK_EN);
  69. i2c_master_write_byte(cmd, reg_addr, ACK_CHECK_EN);
  70. i2c_master_start(cmd);
  71. i2c_master_write_byte(cmd, (slave_addr << 1) | I2C_MASTER_READ, ACK_CHECK_EN);
  72. if (size > 1) {
  73. i2c_master_read(cmd, data_rd, size - 1, ACK_VAL);
  74. }
  75. i2c_master_read_byte(cmd, data_rd + size - 1, NACK_VAL);
  76. i2c_master_stop(cmd);
  77. esp_err_t ret = i2c_master_cmd_begin(i2c_num, cmd, ticks_to_wait);
  78. i2c_cmd_link_delete(cmd);
  79. return ret;
  80. }
  81. /**
  82. * @brief 向SHT3x发送一条指令(16bit)
  83. *
  84. * @param cmd —— SHT3x指令(在SHT3x_MODE中枚举定义)
  85. *
  86. * @retval 成功返回HAL_OK(ESP_OK)
  87. */
  88. static uint8_t SHT3X_Send_Cmd(SHT30_CMD cmd) {
  89. uint8_t cmd_buffer[2];
  90. cmd_buffer[0] = cmd >> 8;
  91. cmd_buffer[1] = cmd;
  92. return i2c_master_write_slave_reg(SHT3X_I2C_BUS, SHT3X_SLAVE_ADDRESS, cmd_buffer[0], cmd_buffer + 1, 1,
  93. SHT3X_TICKS_TO_WAIT);
  94. }
  95. void sht3x_reset(void) {
  96. SHT3X_Send_Cmd(SOFT_RESET_CMD);
  97. vTaskDelay(20 / portTICK_PERIOD_MS);
  98. }
  99. esp_err_t sht3x_init(void) {
  100. return SHT3X_Send_Cmd(MEDIUM_2_CMD);
  101. }
  102. esp_err_t sht3x_read_th_raw_dat(uint8_t *dat) {
  103. return i2c_master_read_slave_reg_16bit(SHT3X_I2C_BUS, SHT3X_SLAVE_ADDRESS, READOUT_FOR_PERIODIC_MODE, dat, 6,
  104. SHT3X_TICKS_TO_WAIT);
  105. }
  106. #define CRC8_POLYNOMIAL 0x31
  107. static uint8_t SHT3X_CheckCrc8(uint8_t *const message, uint8_t initial_value) {
  108. uint8_t remainder; //余数
  109. uint8_t i = 0, j = 0; //循环变量
  110. /* 初始化 */
  111. remainder = initial_value;
  112. for (j = 0; j < 2; j++) {
  113. remainder ^= message[j];
  114. /* 从最高位开始依次计算 */
  115. for (i = 0; i < 8; i++) {
  116. if (remainder & 0x80) {
  117. remainder = (remainder << 1) ^ CRC8_POLYNOMIAL;
  118. } else {
  119. remainder = (remainder << 1);
  120. }
  121. }
  122. }
  123. /* 返回计算的CRC码 */
  124. return remainder;
  125. }
  126. /**
  127. * @brief 将SHT30接收的6个字节数据进行CRC校验,并转换为温度值和湿度值
  128. *
  129. * @param dat —— 存储接收数据的地址(6个字节数组)
  130. *
  131. * @retval 校验成功 —— 返回0
  132. * 校验失败 —— 返回1,并设置温度值和湿度值为0
  133. */
  134. uint8_t sht3x_dat2float(uint8_t *const dat, float *temperature, float *humidity) {
  135. uint16_t recv_temperature = 0;
  136. uint16_t recv_humidity = 0;
  137. /* 校验温度数据和湿度数据是否接收正确 */
  138. if (SHT3X_CheckCrc8(dat, 0xFF) != dat[2] || SHT3X_CheckCrc8(&dat[3], 0xFF) != dat[5])
  139. return 1;
  140. /* 转换温度数据 */
  141. recv_temperature = ((uint16_t) dat[0] << 8) | dat[1];
  142. *temperature = -45 + 175 * ((float) recv_temperature / 65535) + system_setting.fix_temp;
  143. /* 转换湿度数据 */
  144. recv_humidity = ((uint16_t) dat[3] << 8) | dat[4];
  145. *humidity = 100 * ((float) recv_humidity / 65535) + system_setting.fix_hum;
  146. return 0;
  147. }
  148. LV_IMAGE_DECLARE(_wifi_mid_RGB565A8_32x32);
  149. LV_IMAGE_DECLARE(_wifi_weak_RGB565A8_32x32);
  150. void ui_wifi_lab_refresh() {
  151. int rssi = 0;
  152. if (get_wifi_status()) {
  153. esp_err_t ret = esp_wifi_sta_get_rssi(&rssi);
  154. ESP_LOGE(TAG, "RSSI:%d", rssi);
  155. if (ret == ESP_OK) {
  156. lvgl_port_lock(-1);
  157. if (rssi >= -50) {
  158. lv_image_set_src(guider_ui.screen_img_wifi, &_wifi_stron_RGB565A8_32x32);
  159. } else if ((rssi < -50) && (rssi >= -70)) {
  160. lv_image_set_src(guider_ui.screen_img_wifi, &_wifi_mid_RGB565A8_32x32);
  161. } else if (rssi < -70) {
  162. lv_image_set_src(guider_ui.screen_img_wifi, &_wifi_weak_RGB565A8_32x32);
  163. }
  164. lvgl_port_unlock();
  165. }
  166. } else {
  167. }
  168. }
  169. _Noreturn void i2c_sht3x_task(void *arg) {
  170. // 配置I2C0-主机模式,400K,指定 SCL-5,SDA-4
  171. //ESP_ERROR_CHECK(i2c_master_init());
  172. uint8_t recv_dat[6] = {0};
  173. ESP_LOGI(TAG, "esp32 sht3x project starting ……");
  174. sht3x_reset(); // 复位SHT3X
  175. if (sht3x_init() == ESP_OK) // 初始化SHT3X(周期测量模式)
  176. ESP_LOGI(TAG, "sht3x init ok.\n");
  177. else
  178. ESP_LOGE(TAG, "sht3x init fail.\n");
  179. vTaskDelay(1000 / portTICK_PERIOD_MS); //延时1s 等待SHT3X传感器内部采样完成
  180. char temp_buf[32];
  181. char hum_buf[32];
  182. for (;;) {
  183. if (sht3x_read_th_raw_dat(recv_dat) == ESP_OK) // 从SHT3X读取一次数据(周期测量模式下)
  184. {
  185. // 将SHT3X接收的6个字节数据进行CRC校验,并转换为温度值和湿度值
  186. if (sht3x_dat2float(recv_dat, &sht30Data.temperature, &sht30Data.humidity) == 0) {
  187. sht30Data.temperature = correct_temperature(sht30Data.temperature);
  188. snprintf(temp_buf, sizeof(temp_buf), "%.1f °C", sht30Data.temperature);
  189. snprintf(hum_buf, sizeof(hum_buf), "%.1f %%", sht30Data.humidity);
  190. if (guider_ui.screen) {
  191. lvgl_port_lock(-1);
  192. //lv_label_set_text_static(guider_ui.screen_lab_env_temp_2, temp_buf);
  193. //lv_label_set_text_static(guider_ui.ACPage_lab_env_hum_2, hum_buf);
  194. lv_label_set_text_static(guider_ui.screen_lab_env_temp, temp_buf);
  195. lv_label_set_text_static(guider_ui.screen_lab_env_hum, hum_buf);
  196. lvgl_port_unlock();
  197. }
  198. } else {
  199. ESP_LOGE(TAG, "crc check fail.\n");
  200. }
  201. } else {
  202. ESP_LOGE(TAG, "read data from sht3x fail.\n");
  203. }
  204. if (lcd_st7701_get_blacklight() == false) // 如果当前背光引脚为低电平
  205. {
  206. current_seconds -= 3;
  207. if (current_seconds < 0) current_seconds = 0;
  208. } else
  209. current_seconds += 3;
  210. ui_wifi_lab_refresh();
  211. vTaskDelay(15000 / portTICK_PERIOD_MS);
  212. }
  213. }
  214. void sht30_init() {
  215. xTaskCreate(i2c_sht3x_task, "i2c_sht3x_task", 6 * 1024, NULL, 3, NULL);
  216. }