jiuchuan_dev_board.cc 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301
  1. #include "wifi_board.h"
  2. #include "codecs/es8311_audio_codec.h"
  3. #include "display/lcd_display.h"
  4. #include "application.h"
  5. #include "button.h"
  6. #include "config.h"
  7. #include "i2c_device.h"
  8. #include <esp_log.h>
  9. #include <esp_lcd_panel_vendor.h>
  10. #include <driver/i2c_master.h>
  11. #include <driver/spi_common.h>
  12. #include <wifi_station.h>
  13. #include "led/single_led.h"
  14. #include "assets/lang_config.h"
  15. #include "esp_lcd_panel_gc9301.h"
  16. #include "power_save_timer.h"
  17. #include "power_manager.h"
  18. #include "power_controller.h"
  19. #include "gpio_manager.h"
  20. #include <driver/rtc_io.h>
  21. #include <esp_sleep.h>
  22. #define TAG "JiuchuanDevBoard"
  23. #define __USER_GPIO_PWRDOWN__
  24. LV_FONT_DECLARE(font_puhui_20_4);
  25. LV_FONT_DECLARE(font_awesome_20_4);
  26. class JiuchuanDevBoard : public WifiBoard {
  27. private:
  28. i2c_master_bus_handle_t codec_i2c_bus_;
  29. Button boot_button_;
  30. Button pwr_button_;
  31. Button wifi_button;
  32. Button cmd_button;
  33. LcdDisplay* display_;
  34. PowerSaveTimer* power_save_timer_;
  35. PowerManager* power_manager_;
  36. esp_lcd_panel_io_handle_t panel_io = NULL;
  37. esp_lcd_panel_handle_t panel = NULL;
  38. void InitializePowerManager() {
  39. power_manager_ = new PowerManager(PWR_ADC_GPIO);
  40. power_manager_->OnChargingStatusChanged([this](bool is_charging) {
  41. if (is_charging) {
  42. power_save_timer_->SetEnabled(false);
  43. } else {
  44. power_save_timer_->SetEnabled(true);
  45. }
  46. });
  47. }
  48. void InitializePowerSaveTimer() {
  49. #ifndef __USER_GPIO_PWRDOWN__
  50. RTC_DATA_ATTR static bool long_press_occurred = false;
  51. esp_sleep_wakeup_cause_t cause = esp_sleep_get_wakeup_cause();
  52. if (cause == ESP_SLEEP_WAKEUP_EXT0) {
  53. ESP_LOGI(TAG, "Wake up by EXT0");
  54. const int64_t start = esp_timer_get_time();
  55. ESP_LOGI(TAG, "esp_sleep_get_wakeup_cause");
  56. while (gpio_get_level(PWR_BUTTON_GPIO) == 0) {
  57. if (esp_timer_get_time() - start > 3000000) {
  58. long_press_occurred = true;
  59. break;
  60. }
  61. vTaskDelay(100 / portTICK_PERIOD_MS);
  62. }
  63. if (long_press_occurred) {
  64. ESP_LOGI(TAG, "Long press wakeup");
  65. long_press_occurred = false;
  66. } else {
  67. ESP_LOGI(TAG, "Short press, return to sleep");
  68. ESP_ERROR_CHECK(esp_sleep_enable_ext0_wakeup(PWR_BUTTON_GPIO, 0));
  69. ESP_ERROR_CHECK(rtc_gpio_pullup_en(PWR_BUTTON_GPIO)); // 内部上拉
  70. ESP_ERROR_CHECK(rtc_gpio_pulldown_dis(PWR_BUTTON_GPIO));
  71. esp_deep_sleep_start();
  72. }
  73. }
  74. #endif
  75. //一分钟进入浅睡眠,5分钟进入深睡眠关机
  76. power_save_timer_ = new PowerSaveTimer(-1, (60*10), (60*30));
  77. // power_save_timer_ = new PowerSaveTimer(-1, 6, 10);//test
  78. power_save_timer_->OnEnterSleepMode([this]() {
  79. ESP_LOGI(TAG, "Enabling sleep mode");
  80. display_->SetChatMessage("system", "");
  81. display_->SetEmotion("sleepy");
  82. GetBacklight()->SetBrightness(1);
  83. });
  84. power_save_timer_->OnExitSleepMode([this]() {
  85. display_->SetChatMessage("system", "");
  86. display_->SetEmotion("neutral");
  87. GetBacklight()->RestoreBrightness();
  88. });
  89. power_save_timer_->OnShutdownRequest([this]() {
  90. ESP_LOGI(TAG, "Shutting down");
  91. #ifndef __USER_GPIO_PWRDOWN__
  92. ESP_ERROR_CHECK(esp_sleep_enable_ext0_wakeup(PWR_BUTTON_GPIO, 0));
  93. ESP_ERROR_CHECK(rtc_gpio_pullup_en(PWR_BUTTON_GPIO)); // 内部上拉
  94. ESP_ERROR_CHECK(rtc_gpio_pulldown_dis(PWR_BUTTON_GPIO));
  95. esp_lcd_panel_disp_on_off(panel, false); //关闭显示
  96. esp_deep_sleep_start();
  97. #else
  98. rtc_gpio_set_level(PWR_EN_GPIO, 0);
  99. rtc_gpio_hold_dis(PWR_EN_GPIO);
  100. #endif
  101. });
  102. power_save_timer_->SetEnabled(true);
  103. }
  104. void InitializeI2c() {
  105. // Initialize I2C peripheral
  106. i2c_master_bus_config_t i2c_bus_cfg = {
  107. .i2c_port = (i2c_port_t)1,
  108. .sda_io_num = AUDIO_CODEC_I2C_SDA_PIN,
  109. .scl_io_num = AUDIO_CODEC_I2C_SCL_PIN,
  110. .clk_source = I2C_CLK_SRC_DEFAULT,
  111. .glitch_ignore_cnt = 7,
  112. .intr_priority = 0,
  113. .trans_queue_depth = 0,
  114. .flags = {
  115. .enable_internal_pullup = 1,
  116. },
  117. };
  118. ESP_ERROR_CHECK(i2c_new_master_bus(&i2c_bus_cfg, &codec_i2c_bus_));
  119. }
  120. void InitializeButtons() {
  121. // 配置GPIO
  122. ESP_LOGI(TAG, "Configuring power button GPIO");
  123. GpioManager::Config(GPIO_NUM_3, GpioManager::GpioMode::INPUT_PULLDOWN);
  124. boot_button_.OnClick([this]() {
  125. ESP_LOGI(TAG, "Boot button clicked");
  126. power_save_timer_->WakeUp();
  127. });
  128. // 检查电源按钮初始状态
  129. ESP_LOGI(TAG, "Power button initial state: %d", GpioManager::GetLevel(PWR_BUTTON_GPIO));
  130. // 高电平有效长按关机逻辑
  131. pwr_button_.OnLongPress([this]() {
  132. ESP_LOGI(TAG, "Power button long press detected (high-active)");
  133. // 高电平有效防抖确认
  134. for (int i = 0; i < 5; i++) {
  135. int level = GpioManager::GetLevel(PWR_BUTTON_GPIO);
  136. ESP_LOGD(TAG, "Debounce check %d: GPIO%d level=%d", i+1, PWR_BUTTON_GPIO, level);
  137. if (level == 0) {
  138. ESP_LOGW(TAG, "Power button inactive during confirmation - abort shutdown");
  139. return;
  140. }
  141. vTaskDelay(100 / portTICK_PERIOD_MS);
  142. }
  143. ESP_LOGI(TAG, "Confirmed power button pressed - initiating shutdown");
  144. power_manager_->SetPowerState(PowerState::SHUTDOWN);
  145. });
  146. wifi_button.OnClick([this]() {
  147. ESP_LOGI(TAG, "Wifi button clicked");
  148. power_save_timer_->WakeUp();
  149. ESP_LOGI(TAG, "Resetting WiFi configuration");
  150. GpioManager::SetLevel(PWR_EN_GPIO, 1);
  151. ResetWifiConfiguration();
  152. });
  153. cmd_button.OnClick([this]() {
  154. ESP_LOGI(TAG, "Command button clicked");
  155. power_save_timer_->WakeUp();
  156. Application::GetInstance().ToggleChatState();
  157. });
  158. }
  159. void InitializeGC9301isplay() {
  160. // 液晶屏控制IO初始化
  161. ESP_LOGI(TAG, "test Install panel IO");
  162. spi_bus_config_t buscfg = {};
  163. buscfg.mosi_io_num = DISPLAY_SPI_MOSI_PIN;
  164. buscfg.sclk_io_num = DISPLAY_SPI_SCK_PIN;
  165. buscfg.miso_io_num = GPIO_NUM_NC;
  166. buscfg.quadwp_io_num = GPIO_NUM_NC;
  167. buscfg.quadhd_io_num = GPIO_NUM_NC;
  168. buscfg.max_transfer_sz = DISPLAY_WIDTH * DISPLAY_HEIGHT * sizeof(uint16_t);
  169. ESP_ERROR_CHECK(spi_bus_initialize(SPI3_HOST, &buscfg, SPI_DMA_CH_AUTO));
  170. // 初始化SPI总线
  171. esp_lcd_panel_io_spi_config_t io_config = {};
  172. io_config.cs_gpio_num = DISPLAY_SPI_CS_PIN;
  173. io_config.dc_gpio_num = DISPLAY_DC_PIN;
  174. io_config.spi_mode = 3;
  175. io_config.pclk_hz = 80 * 1000 * 1000;
  176. io_config.trans_queue_depth = 10;
  177. io_config.lcd_cmd_bits = 8;
  178. io_config.lcd_param_bits = 8;
  179. esp_lcd_new_panel_io_spi(SPI3_HOST, &io_config, &panel_io);
  180. // 初始化液晶屏驱动芯片9309
  181. ESP_LOGI(TAG, "Install LCD driver");
  182. esp_lcd_panel_dev_config_t panel_config = {};
  183. panel_config.reset_gpio_num = GPIO_NUM_NC;
  184. panel_config.rgb_ele_order = LCD_RGB_ENDIAN_BGR;
  185. panel_config.bits_per_pixel = 16;
  186. esp_lcd_new_panel_gc9309na(panel_io, &panel_config, &panel);
  187. esp_lcd_panel_reset(panel);
  188. esp_lcd_panel_init(panel);
  189. esp_lcd_panel_invert_color(panel, false);
  190. esp_lcd_panel_swap_xy(panel, DISPLAY_SWAP_XY);
  191. esp_lcd_panel_mirror(panel, DISPLAY_MIRROR_X, DISPLAY_MIRROR_Y);
  192. display_ = new SpiLcdDisplay(panel_io, panel,
  193. DISPLAY_WIDTH, DISPLAY_HEIGHT, DISPLAY_OFFSET_X, DISPLAY_OFFSET_Y, DISPLAY_MIRROR_X, DISPLAY_MIRROR_Y, DISPLAY_SWAP_XY,
  194. {
  195. .text_font = &font_puhui_20_4,
  196. .icon_font = &font_awesome_20_4,
  197. #if CONFIG_USE_WECHAT_MESSAGE_STYLE
  198. .emoji_font = font_emoji_32_init(),
  199. #else
  200. .emoji_font = font_emoji_64_init(),
  201. #endif
  202. });
  203. }
  204. public:
  205. JiuchuanDevBoard() :
  206. boot_button_(BOOT_BUTTON_GPIO),
  207. pwr_button_(PWR_BUTTON_GPIO,true),
  208. wifi_button(WIFI_BUTTON_GPIO),
  209. cmd_button(CMD_BUTTON_GPIO) {
  210. InitializeI2c();
  211. InitializePowerManager();
  212. InitializePowerSaveTimer();
  213. InitializeButtons();
  214. InitializeGC9301isplay();
  215. GetBacklight()->RestoreBrightness();
  216. }
  217. virtual Led* GetLed() override {
  218. static SingleLed led(BUILTIN_LED_GPIO);
  219. return &led;
  220. }
  221. virtual AudioCodec* GetAudioCodec() override {
  222. static Es8311AudioCodec audio_codec(
  223. codec_i2c_bus_,
  224. I2C_NUM_0,
  225. AUDIO_INPUT_SAMPLE_RATE,
  226. AUDIO_OUTPUT_SAMPLE_RATE,
  227. AUDIO_I2S_GPIO_MCLK,
  228. AUDIO_I2S_GPIO_BCLK,
  229. AUDIO_I2S_GPIO_WS,
  230. AUDIO_I2S_GPIO_DOUT,
  231. AUDIO_I2S_GPIO_DIN,
  232. AUDIO_CODEC_PA_PIN,
  233. AUDIO_CODEC_ES8311_ADDR);
  234. return &audio_codec;
  235. }
  236. virtual Display* GetDisplay() override {
  237. return display_;
  238. }
  239. virtual Backlight* GetBacklight() override {
  240. static PwmBacklight backlight(DISPLAY_BACKLIGHT_PIN, DISPLAY_BACKLIGHT_OUTPUT_INVERT);
  241. return &backlight;
  242. }
  243. virtual bool GetBatteryLevel(int& level, bool& charging, bool& discharging) override {
  244. static bool last_discharging = false;
  245. charging = power_manager_->IsCharging();
  246. discharging = power_manager_->IsDischarging();
  247. if (discharging != last_discharging) {
  248. power_save_timer_->SetEnabled(discharging);
  249. last_discharging = discharging;
  250. }
  251. level = power_manager_->GetBatteryLevel();
  252. return true;
  253. }
  254. virtual void SetPowerSaveMode(bool enabled) override {
  255. if (!enabled) {
  256. power_save_timer_->WakeUp();
  257. }
  258. WifiBoard::SetPowerSaveMode(enabled);
  259. }
  260. };
  261. DECLARE_BOARD(JiuchuanDevBoard);