EchoEar.cc 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585
  1. #include "wifi_board.h"
  2. #include "codecs/box_audio_codec.h"
  3. #include "display/lcd_display.h"
  4. #include "application.h"
  5. #include "button.h"
  6. #include "config.h"
  7. #include "backlight.h"
  8. #include <wifi_station.h>
  9. #include <esp_log.h>
  10. #include <driver/i2c_master.h>
  11. #include <driver/i2c.h>
  12. #include "i2c_device.h"
  13. #include <esp_lcd_panel_io.h>
  14. #include <esp_lcd_panel_ops.h>
  15. #include <esp_lcd_st77916.h>
  16. #include "esp_lcd_touch_cst816s.h"
  17. #include "touch.h"
  18. #include "driver/temperature_sensor.h"
  19. #include <freertos/FreeRTOS.h>
  20. #include <freertos/semphr.h>
  21. #include <freertos/task.h>
  22. #define TAG "EchoEar"
  23. LV_FONT_DECLARE(font_puhui_20_4);
  24. LV_FONT_DECLARE(font_awesome_20_4);
  25. temperature_sensor_handle_t temp_sensor = NULL;
  26. static const st77916_lcd_init_cmd_t vendor_specific_init_yysj[] = {
  27. {0xF0, (uint8_t []){0x28}, 1, 0},
  28. {0xF2, (uint8_t []){0x28}, 1, 0},
  29. {0x73, (uint8_t []){0xF0}, 1, 0},
  30. {0x7C, (uint8_t []){0xD1}, 1, 0},
  31. {0x83, (uint8_t []){0xE0}, 1, 0},
  32. {0x84, (uint8_t []){0x61}, 1, 0},
  33. {0xF2, (uint8_t []){0x82}, 1, 0},
  34. {0xF0, (uint8_t []){0x00}, 1, 0},
  35. {0xF0, (uint8_t []){0x01}, 1, 0},
  36. {0xF1, (uint8_t []){0x01}, 1, 0},
  37. {0xB0, (uint8_t []){0x56}, 1, 0},
  38. {0xB1, (uint8_t []){0x4D}, 1, 0},
  39. {0xB2, (uint8_t []){0x24}, 1, 0},
  40. {0xB4, (uint8_t []){0x87}, 1, 0},
  41. {0xB5, (uint8_t []){0x44}, 1, 0},
  42. {0xB6, (uint8_t []){0x8B}, 1, 0},
  43. {0xB7, (uint8_t []){0x40}, 1, 0},
  44. {0xB8, (uint8_t []){0x86}, 1, 0},
  45. {0xBA, (uint8_t []){0x00}, 1, 0},
  46. {0xBB, (uint8_t []){0x08}, 1, 0},
  47. {0xBC, (uint8_t []){0x08}, 1, 0},
  48. {0xBD, (uint8_t []){0x00}, 1, 0},
  49. {0xC0, (uint8_t []){0x80}, 1, 0},
  50. {0xC1, (uint8_t []){0x10}, 1, 0},
  51. {0xC2, (uint8_t []){0x37}, 1, 0},
  52. {0xC3, (uint8_t []){0x80}, 1, 0},
  53. {0xC4, (uint8_t []){0x10}, 1, 0},
  54. {0xC5, (uint8_t []){0x37}, 1, 0},
  55. {0xC6, (uint8_t []){0xA9}, 1, 0},
  56. {0xC7, (uint8_t []){0x41}, 1, 0},
  57. {0xC8, (uint8_t []){0x01}, 1, 0},
  58. {0xC9, (uint8_t []){0xA9}, 1, 0},
  59. {0xCA, (uint8_t []){0x41}, 1, 0},
  60. {0xCB, (uint8_t []){0x01}, 1, 0},
  61. {0xD0, (uint8_t []){0x91}, 1, 0},
  62. {0xD1, (uint8_t []){0x68}, 1, 0},
  63. {0xD2, (uint8_t []){0x68}, 1, 0},
  64. {0xF5, (uint8_t []){0x00, 0xA5}, 2, 0},
  65. {0xDD, (uint8_t []){0x4F}, 1, 0},
  66. {0xDE, (uint8_t []){0x4F}, 1, 0},
  67. {0xF1, (uint8_t []){0x10}, 1, 0},
  68. {0xF0, (uint8_t []){0x00}, 1, 0},
  69. {0xF0, (uint8_t []){0x02}, 1, 0},
  70. {0xE0, (uint8_t []){0xF0, 0x0A, 0x10, 0x09, 0x09, 0x36, 0x35, 0x33, 0x4A, 0x29, 0x15, 0x15, 0x2E, 0x34}, 14, 0},
  71. {0xE1, (uint8_t []){0xF0, 0x0A, 0x0F, 0x08, 0x08, 0x05, 0x34, 0x33, 0x4A, 0x39, 0x15, 0x15, 0x2D, 0x33}, 14, 0},
  72. {0xF0, (uint8_t []){0x10}, 1, 0},
  73. {0xF3, (uint8_t []){0x10}, 1, 0},
  74. {0xE0, (uint8_t []){0x07}, 1, 0},
  75. {0xE1, (uint8_t []){0x00}, 1, 0},
  76. {0xE2, (uint8_t []){0x00}, 1, 0},
  77. {0xE3, (uint8_t []){0x00}, 1, 0},
  78. {0xE4, (uint8_t []){0xE0}, 1, 0},
  79. {0xE5, (uint8_t []){0x06}, 1, 0},
  80. {0xE6, (uint8_t []){0x21}, 1, 0},
  81. {0xE7, (uint8_t []){0x01}, 1, 0},
  82. {0xE8, (uint8_t []){0x05}, 1, 0},
  83. {0xE9, (uint8_t []){0x02}, 1, 0},
  84. {0xEA, (uint8_t []){0xDA}, 1, 0},
  85. {0xEB, (uint8_t []){0x00}, 1, 0},
  86. {0xEC, (uint8_t []){0x00}, 1, 0},
  87. {0xED, (uint8_t []){0x0F}, 1, 0},
  88. {0xEE, (uint8_t []){0x00}, 1, 0},
  89. {0xEF, (uint8_t []){0x00}, 1, 0},
  90. {0xF8, (uint8_t []){0x00}, 1, 0},
  91. {0xF9, (uint8_t []){0x00}, 1, 0},
  92. {0xFA, (uint8_t []){0x00}, 1, 0},
  93. {0xFB, (uint8_t []){0x00}, 1, 0},
  94. {0xFC, (uint8_t []){0x00}, 1, 0},
  95. {0xFD, (uint8_t []){0x00}, 1, 0},
  96. {0xFE, (uint8_t []){0x00}, 1, 0},
  97. {0xFF, (uint8_t []){0x00}, 1, 0},
  98. {0x60, (uint8_t []){0x40}, 1, 0},
  99. {0x61, (uint8_t []){0x04}, 1, 0},
  100. {0x62, (uint8_t []){0x00}, 1, 0},
  101. {0x63, (uint8_t []){0x42}, 1, 0},
  102. {0x64, (uint8_t []){0xD9}, 1, 0},
  103. {0x65, (uint8_t []){0x00}, 1, 0},
  104. {0x66, (uint8_t []){0x00}, 1, 0},
  105. {0x67, (uint8_t []){0x00}, 1, 0},
  106. {0x68, (uint8_t []){0x00}, 1, 0},
  107. {0x69, (uint8_t []){0x00}, 1, 0},
  108. {0x6A, (uint8_t []){0x00}, 1, 0},
  109. {0x6B, (uint8_t []){0x00}, 1, 0},
  110. {0x70, (uint8_t []){0x40}, 1, 0},
  111. {0x71, (uint8_t []){0x03}, 1, 0},
  112. {0x72, (uint8_t []){0x00}, 1, 0},
  113. {0x73, (uint8_t []){0x42}, 1, 0},
  114. {0x74, (uint8_t []){0xD8}, 1, 0},
  115. {0x75, (uint8_t []){0x00}, 1, 0},
  116. {0x76, (uint8_t []){0x00}, 1, 0},
  117. {0x77, (uint8_t []){0x00}, 1, 0},
  118. {0x78, (uint8_t []){0x00}, 1, 0},
  119. {0x79, (uint8_t []){0x00}, 1, 0},
  120. {0x7A, (uint8_t []){0x00}, 1, 0},
  121. {0x7B, (uint8_t []){0x00}, 1, 0},
  122. {0x80, (uint8_t []){0x48}, 1, 0},
  123. {0x81, (uint8_t []){0x00}, 1, 0},
  124. {0x82, (uint8_t []){0x06}, 1, 0},
  125. {0x83, (uint8_t []){0x02}, 1, 0},
  126. {0x84, (uint8_t []){0xD6}, 1, 0},
  127. {0x85, (uint8_t []){0x04}, 1, 0},
  128. {0x86, (uint8_t []){0x00}, 1, 0},
  129. {0x87, (uint8_t []){0x00}, 1, 0},
  130. {0x88, (uint8_t []){0x48}, 1, 0},
  131. {0x89, (uint8_t []){0x00}, 1, 0},
  132. {0x8A, (uint8_t []){0x08}, 1, 0},
  133. {0x8B, (uint8_t []){0x02}, 1, 0},
  134. {0x8C, (uint8_t []){0xD8}, 1, 0},
  135. {0x8D, (uint8_t []){0x04}, 1, 0},
  136. {0x8E, (uint8_t []){0x00}, 1, 0},
  137. {0x8F, (uint8_t []){0x00}, 1, 0},
  138. {0x90, (uint8_t []){0x48}, 1, 0},
  139. {0x91, (uint8_t []){0x00}, 1, 0},
  140. {0x92, (uint8_t []){0x0A}, 1, 0},
  141. {0x93, (uint8_t []){0x02}, 1, 0},
  142. {0x94, (uint8_t []){0xDA}, 1, 0},
  143. {0x95, (uint8_t []){0x04}, 1, 0},
  144. {0x96, (uint8_t []){0x00}, 1, 0},
  145. {0x97, (uint8_t []){0x00}, 1, 0},
  146. {0x98, (uint8_t []){0x48}, 1, 0},
  147. {0x99, (uint8_t []){0x00}, 1, 0},
  148. {0x9A, (uint8_t []){0x0C}, 1, 0},
  149. {0x9B, (uint8_t []){0x02}, 1, 0},
  150. {0x9C, (uint8_t []){0xDC}, 1, 0},
  151. {0x9D, (uint8_t []){0x04}, 1, 0},
  152. {0x9E, (uint8_t []){0x00}, 1, 0},
  153. {0x9F, (uint8_t []){0x00}, 1, 0},
  154. {0xA0, (uint8_t []){0x48}, 1, 0},
  155. {0xA1, (uint8_t []){0x00}, 1, 0},
  156. {0xA2, (uint8_t []){0x05}, 1, 0},
  157. {0xA3, (uint8_t []){0x02}, 1, 0},
  158. {0xA4, (uint8_t []){0xD5}, 1, 0},
  159. {0xA5, (uint8_t []){0x04}, 1, 0},
  160. {0xA6, (uint8_t []){0x00}, 1, 0},
  161. {0xA7, (uint8_t []){0x00}, 1, 0},
  162. {0xA8, (uint8_t []){0x48}, 1, 0},
  163. {0xA9, (uint8_t []){0x00}, 1, 0},
  164. {0xAA, (uint8_t []){0x07}, 1, 0},
  165. {0xAB, (uint8_t []){0x02}, 1, 0},
  166. {0xAC, (uint8_t []){0xD7}, 1, 0},
  167. {0xAD, (uint8_t []){0x04}, 1, 0},
  168. {0xAE, (uint8_t []){0x00}, 1, 0},
  169. {0xAF, (uint8_t []){0x00}, 1, 0},
  170. {0xB0, (uint8_t []){0x48}, 1, 0},
  171. {0xB1, (uint8_t []){0x00}, 1, 0},
  172. {0xB2, (uint8_t []){0x09}, 1, 0},
  173. {0xB3, (uint8_t []){0x02}, 1, 0},
  174. {0xB4, (uint8_t []){0xD9}, 1, 0},
  175. {0xB5, (uint8_t []){0x04}, 1, 0},
  176. {0xB6, (uint8_t []){0x00}, 1, 0},
  177. {0xB7, (uint8_t []){0x00}, 1, 0},
  178. {0xB8, (uint8_t []){0x48}, 1, 0},
  179. {0xB9, (uint8_t []){0x00}, 1, 0},
  180. {0xBA, (uint8_t []){0x0B}, 1, 0},
  181. {0xBB, (uint8_t []){0x02}, 1, 0},
  182. {0xBC, (uint8_t []){0xDB}, 1, 0},
  183. {0xBD, (uint8_t []){0x04}, 1, 0},
  184. {0xBE, (uint8_t []){0x00}, 1, 0},
  185. {0xBF, (uint8_t []){0x00}, 1, 0},
  186. {0xC0, (uint8_t []){0x10}, 1, 0},
  187. {0xC1, (uint8_t []){0x47}, 1, 0},
  188. {0xC2, (uint8_t []){0x56}, 1, 0},
  189. {0xC3, (uint8_t []){0x65}, 1, 0},
  190. {0xC4, (uint8_t []){0x74}, 1, 0},
  191. {0xC5, (uint8_t []){0x88}, 1, 0},
  192. {0xC6, (uint8_t []){0x99}, 1, 0},
  193. {0xC7, (uint8_t []){0x01}, 1, 0},
  194. {0xC8, (uint8_t []){0xBB}, 1, 0},
  195. {0xC9, (uint8_t []){0xAA}, 1, 0},
  196. {0xD0, (uint8_t []){0x10}, 1, 0},
  197. {0xD1, (uint8_t []){0x47}, 1, 0},
  198. {0xD2, (uint8_t []){0x56}, 1, 0},
  199. {0xD3, (uint8_t []){0x65}, 1, 0},
  200. {0xD4, (uint8_t []){0x74}, 1, 0},
  201. {0xD5, (uint8_t []){0x88}, 1, 0},
  202. {0xD6, (uint8_t []){0x99}, 1, 0},
  203. {0xD7, (uint8_t []){0x01}, 1, 0},
  204. {0xD8, (uint8_t []){0xBB}, 1, 0},
  205. {0xD9, (uint8_t []){0xAA}, 1, 0},
  206. {0xF3, (uint8_t []){0x01}, 1, 0},
  207. {0xF0, (uint8_t []){0x00}, 1, 0},
  208. {0x21, (uint8_t []){}, 0, 0},
  209. {0x11, (uint8_t []){}, 0, 0},
  210. {0x00, (uint8_t []){}, 0, 120},
  211. };
  212. float tsens_value;
  213. gpio_num_t AUDIO_I2S_GPIO_DIN = AUDIO_I2S_GPIO_DIN_1;
  214. gpio_num_t AUDIO_CODEC_PA_PIN = AUDIO_CODEC_PA_PIN_1;
  215. gpio_num_t QSPI_PIN_NUM_LCD_RST = QSPI_PIN_NUM_LCD_RST_1;
  216. gpio_num_t TOUCH_PAD2 = TOUCH_PAD2_1;
  217. gpio_num_t UART1_TX = UART1_TX_1;
  218. gpio_num_t UART1_RX = UART1_RX_1;
  219. class Charge : public I2cDevice {
  220. public:
  221. Charge(i2c_master_bus_handle_t i2c_bus, uint8_t addr) : I2cDevice(i2c_bus, addr) {
  222. read_buffer_ = new uint8_t[8];
  223. }
  224. ~Charge() {
  225. delete[] read_buffer_;
  226. }
  227. void Printcharge() {
  228. ReadRegs(0x08, read_buffer_, 2);
  229. ReadRegs(0x0c, read_buffer_ + 2, 2);
  230. ESP_ERROR_CHECK(temperature_sensor_get_celsius(temp_sensor, &tsens_value));
  231. int16_t voltage = (uint16_t)(read_buffer_[1] << 8 | read_buffer_[0]);
  232. int16_t current = (int16_t)(read_buffer_[3] << 8 | read_buffer_[2]);
  233. }
  234. static void TaskFunction(void *pvParameters) {
  235. Charge* charge = static_cast<Charge*>(pvParameters);
  236. while (true) {
  237. charge->Printcharge();
  238. vTaskDelay(pdMS_TO_TICKS(300));
  239. }
  240. }
  241. private:
  242. uint8_t* read_buffer_ = nullptr;
  243. };
  244. class Cst816s : public I2cDevice {
  245. public:
  246. struct TouchPoint_t {
  247. int num = 0;
  248. int x = -1;
  249. int y = -1;
  250. };
  251. Cst816s(i2c_master_bus_handle_t i2c_bus, uint8_t addr) : I2cDevice(i2c_bus, addr) {
  252. read_buffer_ = new uint8_t[6];
  253. }
  254. ~Cst816s() {
  255. delete[] read_buffer_;
  256. }
  257. void UpdateTouchPoint() {
  258. ReadRegs(0x02, read_buffer_, 6);
  259. tp_.num = read_buffer_[0] & 0x0F;
  260. tp_.x = ((read_buffer_[1] & 0x0F) << 8) | read_buffer_[2];
  261. tp_.y = ((read_buffer_[3] & 0x0F) << 8) | read_buffer_[4];
  262. }
  263. const TouchPoint_t& GetTouchPoint() {
  264. return tp_;
  265. }
  266. private:
  267. uint8_t* read_buffer_ = nullptr;
  268. TouchPoint_t tp_;
  269. };
  270. static SemaphoreHandle_t touch_isr_mux = NULL;
  271. static bool touch_event_pending = false;
  272. static int64_t touch_event_time = 0;
  273. class EspS3Cat : public WifiBoard {
  274. private:
  275. i2c_master_bus_handle_t i2c_bus_;
  276. Cst816s* cst816s_;
  277. Charge* charge_;
  278. Button boot_button_;
  279. LcdDisplay* display_;
  280. PwmBacklight* backlight_ = nullptr;
  281. esp_timer_handle_t touchpad_timer_;
  282. esp_lcd_touch_handle_t tp; // LCD touch handle
  283. void InitializeI2c() {
  284. i2c_master_bus_config_t i2c_bus_cfg = {
  285. .i2c_port = I2C_NUM_0,
  286. .sda_io_num = AUDIO_CODEC_I2C_SDA_PIN,
  287. .scl_io_num = AUDIO_CODEC_I2C_SCL_PIN,
  288. .clk_source = I2C_CLK_SRC_DEFAULT,
  289. .glitch_ignore_cnt = 7,
  290. .intr_priority = 0,
  291. .trans_queue_depth = 0,
  292. .flags = {
  293. .enable_internal_pullup = 1,
  294. },
  295. };
  296. ESP_ERROR_CHECK(i2c_new_master_bus(&i2c_bus_cfg, &i2c_bus_));
  297. temperature_sensor_config_t temp_sensor_config = TEMPERATURE_SENSOR_CONFIG_DEFAULT(10, 50);
  298. ESP_ERROR_CHECK(temperature_sensor_install(&temp_sensor_config, &temp_sensor));
  299. ESP_ERROR_CHECK(temperature_sensor_enable(temp_sensor));
  300. }
  301. uint8_t DetectPcbVersion() {
  302. esp_err_t ret = i2c_master_probe(i2c_bus_, 0x18, 100);
  303. uint8_t pcb_verison = 0;
  304. if (ret == ESP_OK) {
  305. ESP_LOGI(TAG, "PCB verison V1.0");
  306. pcb_verison = 0;
  307. } else {
  308. gpio_config_t gpio_conf = {
  309. .pin_bit_mask = (1ULL << GPIO_NUM_48),
  310. .mode = GPIO_MODE_OUTPUT,
  311. .pull_up_en = GPIO_PULLUP_DISABLE,
  312. .pull_down_en = GPIO_PULLDOWN_DISABLE,
  313. .intr_type = GPIO_INTR_DISABLE
  314. };
  315. ESP_ERROR_CHECK(gpio_config(&gpio_conf));
  316. ESP_ERROR_CHECK(gpio_set_level(GPIO_NUM_48, 1));
  317. vTaskDelay(pdMS_TO_TICKS(100));
  318. ret = i2c_master_probe(i2c_bus_, 0x18, 100);
  319. if (ret == ESP_OK) {
  320. ESP_LOGI(TAG, "PCB verison V1.2");
  321. pcb_verison = 1;
  322. AUDIO_I2S_GPIO_DIN = AUDIO_I2S_GPIO_DIN_2;
  323. AUDIO_CODEC_PA_PIN = AUDIO_CODEC_PA_PIN_2;
  324. QSPI_PIN_NUM_LCD_RST = QSPI_PIN_NUM_LCD_RST_2;
  325. TOUCH_PAD2 = TOUCH_PAD2_2;
  326. UART1_TX = UART1_TX_2;
  327. UART1_RX = UART1_RX_2;
  328. } else {
  329. ESP_LOGE(TAG, "PCB version detection error");
  330. }
  331. }
  332. return pcb_verison;
  333. }
  334. static void touchpad_timer_callback(void* arg) {
  335. auto& board = (EspS3Cat&)Board::GetInstance();
  336. auto touchpad = board.GetTouchpad();
  337. static bool was_touched = false;
  338. static int64_t touch_start_time = 0;
  339. const int64_t TOUCH_THRESHOLD_MS = 500;
  340. touchpad->UpdateTouchPoint();
  341. auto touch_point = touchpad->GetTouchPoint();
  342. if (touch_point.num > 0 && !was_touched) {
  343. was_touched = true;
  344. touch_start_time = esp_timer_get_time() / 1000;
  345. }
  346. else if (touch_point.num == 0 && was_touched) {
  347. was_touched = false;
  348. int64_t touch_duration = (esp_timer_get_time() / 1000) - touch_start_time;
  349. if (touch_duration < TOUCH_THRESHOLD_MS) {
  350. auto& app = Application::GetInstance();
  351. if (app.GetDeviceState() == kDeviceStateStarting &&
  352. !WifiStation::GetInstance().IsConnected()) {
  353. board.ResetWifiConfiguration();
  354. }
  355. app.ToggleChatState();
  356. }
  357. }
  358. }
  359. static void touchpad_callback(Cst816s::TouchPoint_t touch_point) {
  360. auto& board = (EspS3Cat&)Board::GetInstance();
  361. static bool was_touched = false;
  362. static int64_t touch_start_time = 0;
  363. const int64_t TOUCH_THRESHOLD_MS = 500;
  364. if (touch_point.num > 0 && !was_touched) {
  365. was_touched = true;
  366. touch_start_time = esp_timer_get_time() / 1000;
  367. }
  368. else if (touch_point.num == 0 && was_touched) {
  369. was_touched = false;
  370. int64_t touch_duration = (esp_timer_get_time() / 1000) - touch_start_time;
  371. if (touch_duration < TOUCH_THRESHOLD_MS) {
  372. auto& app = Application::GetInstance();
  373. if (app.GetDeviceState() == kDeviceStateStarting &&
  374. !WifiStation::GetInstance().IsConnected()) {
  375. board.ResetWifiConfiguration();
  376. }
  377. app.ToggleChatState();
  378. }
  379. }
  380. }
  381. static void lvgl_port_touch_isr_cb(void* arg)
  382. {
  383. int64_t current_time = esp_timer_get_time() / 1000;
  384. static int64_t last_touch_time = 0;
  385. if (current_time - last_touch_time >= 300) {
  386. touch_event_pending = true;
  387. touch_event_time = current_time;
  388. last_touch_time = current_time;
  389. BaseType_t xHigherPriorityTaskWoken = pdFALSE;
  390. if (touch_isr_mux != NULL) {
  391. xSemaphoreGiveFromISR(touch_isr_mux, &xHigherPriorityTaskWoken);
  392. portYIELD_FROM_ISR(xHigherPriorityTaskWoken);
  393. }
  394. }
  395. }
  396. static void touch_event_task(void* arg) {
  397. while (true) {
  398. if (xSemaphoreTake(touch_isr_mux, portMAX_DELAY) == pdTRUE) {
  399. if (touch_event_pending) {
  400. touch_event_pending = false;
  401. auto& board = (EspS3Cat&)Board::GetInstance();
  402. auto& app = Application::GetInstance();
  403. if (app.GetDeviceState() == kDeviceStateStarting &&
  404. !WifiStation::GetInstance().IsConnected()) {
  405. board.ResetWifiConfiguration();
  406. }
  407. app.ToggleChatState();
  408. }
  409. }
  410. }
  411. }
  412. void InitializeCharge() {
  413. charge_ = new Charge(i2c_bus_, 0x55);
  414. xTaskCreatePinnedToCore(Charge::TaskFunction, "batterydecTask", 3 * 1024, charge_, 6, NULL, 0);
  415. }
  416. void InitializeCst816sTouchPad() {
  417. cst816s_ = new Cst816s(i2c_bus_, 0x15);
  418. touch_isr_mux = xSemaphoreCreateBinary();
  419. if (touch_isr_mux == NULL) {
  420. ESP_LOGE(TAG, "Failed to create touch semaphore");
  421. return;
  422. }
  423. xTaskCreatePinnedToCore(touch_event_task, "touch_task", 4 * 1024, NULL, 5, NULL, 1);
  424. const gpio_config_t int_gpio_config = {
  425. .pin_bit_mask = (1ULL << TP_PIN_NUM_INT),
  426. .mode = GPIO_MODE_INPUT,
  427. .intr_type = GPIO_INTR_NEGEDGE
  428. };
  429. gpio_config(&int_gpio_config);
  430. gpio_install_isr_service(0);
  431. gpio_intr_enable(TP_PIN_NUM_INT);
  432. gpio_isr_handler_add(TP_PIN_NUM_INT, EspS3Cat::lvgl_port_touch_isr_cb, NULL);
  433. }
  434. void InitializeSpi() {
  435. const spi_bus_config_t bus_config = TAIJIPI_ST77916_PANEL_BUS_QSPI_CONFIG(QSPI_PIN_NUM_LCD_PCLK,
  436. QSPI_PIN_NUM_LCD_DATA0,
  437. QSPI_PIN_NUM_LCD_DATA1,
  438. QSPI_PIN_NUM_LCD_DATA2,
  439. QSPI_PIN_NUM_LCD_DATA3,
  440. QSPI_LCD_H_RES * 80 * sizeof(uint16_t));
  441. ESP_ERROR_CHECK(spi_bus_initialize(QSPI_LCD_HOST, &bus_config, SPI_DMA_CH_AUTO));
  442. }
  443. void Initializest77916Display(uint8_t pcb_verison) {
  444. esp_lcd_panel_io_handle_t panel_io = nullptr;
  445. esp_lcd_panel_handle_t panel = nullptr;
  446. const esp_lcd_panel_io_spi_config_t io_config = ST77916_PANEL_IO_QSPI_CONFIG(QSPI_PIN_NUM_LCD_CS, NULL, NULL);
  447. ESP_ERROR_CHECK(esp_lcd_new_panel_io_spi((esp_lcd_spi_bus_handle_t)QSPI_LCD_HOST, &io_config, &panel_io));
  448. st77916_vendor_config_t vendor_config = {
  449. .init_cmds = vendor_specific_init_yysj,
  450. .init_cmds_size = sizeof(vendor_specific_init_yysj) / sizeof(st77916_lcd_init_cmd_t),
  451. .flags = {
  452. .use_qspi_interface = 1,
  453. },
  454. };
  455. const esp_lcd_panel_dev_config_t panel_config = {
  456. .reset_gpio_num = QSPI_PIN_NUM_LCD_RST,
  457. .rgb_ele_order = LCD_RGB_ELEMENT_ORDER_RGB,
  458. .bits_per_pixel = QSPI_LCD_BIT_PER_PIXEL,
  459. .flags = {
  460. .reset_active_high = pcb_verison,
  461. },
  462. .vendor_config = &vendor_config,
  463. };
  464. ESP_ERROR_CHECK(esp_lcd_new_panel_st77916(panel_io, &panel_config, &panel));
  465. esp_lcd_panel_reset(panel);
  466. esp_lcd_panel_init(panel);
  467. esp_lcd_panel_disp_on_off(panel, true);
  468. esp_lcd_panel_swap_xy(panel, DISPLAY_SWAP_XY);
  469. esp_lcd_panel_mirror(panel, DISPLAY_MIRROR_X, DISPLAY_MIRROR_Y);
  470. display_ = new SpiLcdDisplay(panel_io, panel,
  471. DISPLAY_WIDTH, DISPLAY_HEIGHT, DISPLAY_OFFSET_X, DISPLAY_OFFSET_Y, DISPLAY_MIRROR_X, DISPLAY_MIRROR_Y, DISPLAY_SWAP_XY,
  472. {
  473. .text_font = &font_puhui_20_4,
  474. .icon_font = &font_awesome_20_4,
  475. .emoji_font = font_emoji_64_init(),
  476. });
  477. backlight_ = new PwmBacklight(DISPLAY_BACKLIGHT_PIN, DISPLAY_BACKLIGHT_OUTPUT_INVERT);
  478. backlight_->RestoreBrightness();
  479. }
  480. void InitializeButtons() {
  481. boot_button_.OnClick([this]() {
  482. auto& app = Application::GetInstance();
  483. if (app.GetDeviceState() == kDeviceStateStarting && !WifiStation::GetInstance().IsConnected()) {
  484. ResetWifiConfiguration();
  485. }
  486. app.ToggleChatState();
  487. });
  488. gpio_config_t power_gpio_config = {
  489. .pin_bit_mask = (BIT64(POWER_CTRL) ),
  490. .mode = GPIO_MODE_OUTPUT,
  491. };
  492. ESP_ERROR_CHECK(gpio_config(&power_gpio_config));
  493. gpio_set_level(POWER_CTRL, 0);
  494. }
  495. public:
  496. EspS3Cat() : boot_button_(BOOT_BUTTON_GPIO) {
  497. InitializeI2c();
  498. uint8_t pcb_verison = DetectPcbVersion();
  499. InitializeCharge();
  500. InitializeCst816sTouchPad();
  501. InitializeSpi();
  502. Initializest77916Display(pcb_verison);
  503. InitializeButtons();
  504. }
  505. virtual AudioCodec* GetAudioCodec() override {
  506. static BoxAudioCodec audio_codec(
  507. i2c_bus_,
  508. AUDIO_INPUT_SAMPLE_RATE,
  509. AUDIO_OUTPUT_SAMPLE_RATE,
  510. AUDIO_I2S_GPIO_MCLK,
  511. AUDIO_I2S_GPIO_BCLK,
  512. AUDIO_I2S_GPIO_WS,
  513. AUDIO_I2S_GPIO_DOUT,
  514. AUDIO_I2S_GPIO_DIN,
  515. AUDIO_CODEC_PA_PIN,
  516. AUDIO_CODEC_ES8311_ADDR,
  517. AUDIO_CODEC_ES7210_ADDR,
  518. AUDIO_INPUT_REFERENCE);
  519. return &audio_codec;
  520. }
  521. virtual Display* GetDisplay() override {
  522. return display_;
  523. }
  524. Cst816s* GetTouchpad() {
  525. return cst816s_;
  526. }
  527. virtual Backlight* GetBacklight() override {
  528. return backlight_;
  529. }
  530. };
  531. DECLARE_BOARD(EspS3Cat);