sensecap_watcher.cc 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334
  1. #include "display/lv_display.h"
  2. #include "misc/lv_event.h"
  3. #include "wifi_board.h"
  4. #include "sensecap_audio_codec.h"
  5. #include "display/lcd_display.h"
  6. #include "font_awesome_symbols.h"
  7. #include "application.h"
  8. #include "button.h"
  9. #include "knob.h"
  10. #include "config.h"
  11. #include "led/single_led.h"
  12. #include "iot/thing_manager.h"
  13. #include "power_save_timer.h"
  14. #include <esp_log.h>
  15. #include "esp_check.h"
  16. #include <esp_lcd_panel_io.h>
  17. #include <esp_lcd_panel_ops.h>
  18. #include <esp_lcd_spd2010.h>
  19. #include <driver/spi_master.h>
  20. #include <driver/i2c_master.h>
  21. #include <driver/spi_common.h>
  22. #include <wifi_station.h>
  23. #include <iot_button.h>
  24. #include <iot_knob.h>
  25. #include <esp_io_expander_tca95xx_16bit.h>
  26. #include <esp_sleep.h>
  27. #define TAG "sensecap_watcher"
  28. LV_FONT_DECLARE(font_puhui_30_4);
  29. LV_FONT_DECLARE(font_awesome_30_4);
  30. class SensecapWatcher : public WifiBoard {
  31. private:
  32. i2c_master_bus_handle_t i2c_bus_;
  33. LcdDisplay* display_;
  34. std::unique_ptr<Knob> knob_;
  35. esp_io_expander_handle_t io_exp_handle;
  36. button_handle_t btns;
  37. PowerSaveTimer* power_save_timer_;
  38. esp_lcd_panel_io_handle_t panel_io_ = nullptr;
  39. esp_lcd_panel_handle_t panel_ = nullptr;
  40. void InitializePowerSaveTimer() {
  41. power_save_timer_ = new PowerSaveTimer(-1, 60, 300);
  42. power_save_timer_->OnEnterSleepMode([this]() {
  43. ESP_LOGI(TAG, "Enabling sleep mode");
  44. auto display = GetDisplay();
  45. display->SetChatMessage("system", "");
  46. display->SetEmotion("sleepy");
  47. GetBacklight()->SetBrightness(10);
  48. });
  49. power_save_timer_->OnExitSleepMode([this]() {
  50. auto display = GetDisplay();
  51. display->SetChatMessage("system", "");
  52. display->SetEmotion("neutral");
  53. GetBacklight()->RestoreBrightness();
  54. });
  55. power_save_timer_->OnShutdownRequest([this]() {
  56. ESP_LOGI(TAG, "Shutting down");
  57. bool is_charging = (IoExpanderGetLevel(BSP_PWR_VBUS_IN_DET) == 0);
  58. if (is_charging) {
  59. ESP_LOGI(TAG, "charging");
  60. GetBacklight()->SetBrightness(0);
  61. } else {
  62. IoExpanderSetLevel(BSP_PWR_SYSTEM, 0);
  63. }
  64. });
  65. power_save_timer_->SetEnabled(true);
  66. }
  67. void InitializeI2c() {
  68. // Initialize I2C peripheral
  69. i2c_master_bus_config_t i2c_bus_cfg = {
  70. .i2c_port = (i2c_port_t)0,
  71. .sda_io_num = BSP_GENERAL_I2C_SDA,
  72. .scl_io_num = BSP_GENERAL_I2C_SCL,
  73. .clk_source = I2C_CLK_SRC_DEFAULT,
  74. .glitch_ignore_cnt = 7,
  75. .intr_priority = 0,
  76. .trans_queue_depth = 0,
  77. .flags = {
  78. .enable_internal_pullup = 1,
  79. },
  80. };
  81. ESP_ERROR_CHECK(i2c_new_master_bus(&i2c_bus_cfg, &i2c_bus_));
  82. }
  83. esp_err_t IoExpanderSetLevel(uint16_t pin_mask, uint8_t level) {
  84. return esp_io_expander_set_level(io_exp_handle, pin_mask, level);
  85. }
  86. uint8_t IoExpanderGetLevel(uint16_t pin_mask) {
  87. uint32_t pin_val = 0;
  88. esp_io_expander_get_level(io_exp_handle, DRV_IO_EXP_INPUT_MASK, &pin_val);
  89. pin_mask &= DRV_IO_EXP_INPUT_MASK;
  90. return (uint8_t)((pin_val & pin_mask) ? 1 : 0);
  91. }
  92. void InitializeExpander() {
  93. esp_err_t ret = ESP_OK;
  94. esp_io_expander_new_i2c_tca95xx_16bit(i2c_bus_, ESP_IO_EXPANDER_I2C_TCA9555_ADDRESS_001, &io_exp_handle);
  95. ret |= esp_io_expander_set_dir(io_exp_handle, DRV_IO_EXP_INPUT_MASK, IO_EXPANDER_INPUT);
  96. ret |= esp_io_expander_set_dir(io_exp_handle, DRV_IO_EXP_OUTPUT_MASK, IO_EXPANDER_OUTPUT);
  97. ret |= esp_io_expander_set_level(io_exp_handle, DRV_IO_EXP_OUTPUT_MASK, 0);
  98. ret |= esp_io_expander_set_level(io_exp_handle, BSP_PWR_SYSTEM, 1);
  99. vTaskDelay(100 / portTICK_PERIOD_MS);
  100. ret |= esp_io_expander_set_level(io_exp_handle, BSP_PWR_START_UP, 1);
  101. vTaskDelay(50 / portTICK_PERIOD_MS);
  102. uint32_t pin_val = 0;
  103. ret |= esp_io_expander_get_level(io_exp_handle, DRV_IO_EXP_INPUT_MASK, &pin_val);
  104. ESP_LOGI(TAG, "IO expander initialized: %x", DRV_IO_EXP_OUTPUT_MASK | (uint16_t)pin_val);
  105. assert(ret == ESP_OK);
  106. }
  107. void OnKnobRotate(bool clockwise) {
  108. auto codec = GetAudioCodec();
  109. int current_volume = codec->output_volume();
  110. int new_volume = current_volume + (clockwise ? 5 : -5);
  111. // 确保音量在有效范围内
  112. if (new_volume > 100) {
  113. new_volume = 100;
  114. ESP_LOGW(TAG, "Volume reached maximum limit: %d", new_volume);
  115. } else if (new_volume < 0) {
  116. new_volume = 0;
  117. ESP_LOGW(TAG, "Volume reached minimum limit: %d", new_volume);
  118. }
  119. codec->SetOutputVolume(new_volume);
  120. ESP_LOGI(TAG, "Volume changed from %d to %d", current_volume, new_volume);
  121. // 显示通知前检查实际变化
  122. if (new_volume != codec->output_volume()) {
  123. ESP_LOGE(TAG, "Failed to set volume! Expected:%d Actual:%d",
  124. new_volume, codec->output_volume());
  125. }
  126. GetDisplay()->ShowNotification("音量: " + std::to_string(codec->output_volume()));
  127. power_save_timer_->WakeUp();
  128. }
  129. void InitializeKnob() {
  130. knob_ = std::make_unique<Knob>(BSP_KNOB_A_PIN, BSP_KNOB_B_PIN);
  131. knob_->OnRotate([this](bool clockwise) {
  132. ESP_LOGD(TAG, "Knob rotation detected. Clockwise:%s", clockwise ? "true" : "false");
  133. OnKnobRotate(clockwise);
  134. });
  135. ESP_LOGI(TAG, "Knob initialized with pins A:%d B:%d", BSP_KNOB_A_PIN, BSP_KNOB_B_PIN);
  136. }
  137. void InitializeButton() {
  138. button_config_t btn_config = {
  139. .type = BUTTON_TYPE_CUSTOM,
  140. .long_press_time = 2000,
  141. .short_press_time = 50,
  142. .custom_button_config = {
  143. .active_level = 0,
  144. .button_custom_init =nullptr,
  145. .button_custom_get_key_value = [](void *param) -> uint8_t {
  146. auto self = static_cast<SensecapWatcher*>(param);
  147. return self->IoExpanderGetLevel(BSP_KNOB_BTN);
  148. },
  149. .button_custom_deinit = nullptr,
  150. .priv = this,
  151. },
  152. };
  153. //watcher 是通过长按滚轮进行开机的, 需要等待滚轮释放, 否则用户开机松手时可能会误触成单击
  154. ESP_LOGI(TAG, "waiting for knob button release");
  155. while(IoExpanderGetLevel(BSP_KNOB_BTN) == 0) {
  156. vTaskDelay(50 / portTICK_PERIOD_MS);
  157. }
  158. btns = iot_button_create(&btn_config);
  159. iot_button_register_cb(btns, BUTTON_SINGLE_CLICK, [](void* button_handle, void* usr_data) {
  160. auto self = static_cast<SensecapWatcher*>(usr_data);
  161. auto& app = Application::GetInstance();
  162. if (app.GetDeviceState() == kDeviceStateStarting && !WifiStation::GetInstance().IsConnected()) {
  163. self->ResetWifiConfiguration();
  164. }
  165. self->power_save_timer_->WakeUp();
  166. app.ToggleChatState();
  167. }, this);
  168. iot_button_register_cb(btns, BUTTON_LONG_PRESS_START, [](void* button_handle, void* usr_data) {
  169. auto self = static_cast<SensecapWatcher*>(usr_data);
  170. bool is_charging = (self->IoExpanderGetLevel(BSP_PWR_VBUS_IN_DET) == 0);
  171. if (is_charging) {
  172. ESP_LOGI(TAG, "charging");
  173. } else {
  174. self->IoExpanderSetLevel(BSP_PWR_LCD, 0);
  175. self->IoExpanderSetLevel(BSP_PWR_SYSTEM, 0);
  176. }
  177. }, this);
  178. }
  179. void InitializeSpi() {
  180. ESP_LOGI(TAG, "Initialize QSPI bus");
  181. spi_bus_config_t qspi_cfg = {0};
  182. qspi_cfg.sclk_io_num = BSP_SPI3_HOST_PCLK;
  183. qspi_cfg.data0_io_num = BSP_SPI3_HOST_DATA0;
  184. qspi_cfg.data1_io_num = BSP_SPI3_HOST_DATA1;
  185. qspi_cfg.data2_io_num = BSP_SPI3_HOST_DATA2;
  186. qspi_cfg.data3_io_num = BSP_SPI3_HOST_DATA3;
  187. qspi_cfg.max_transfer_sz = DISPLAY_WIDTH * DISPLAY_HEIGHT * DRV_LCD_BITS_PER_PIXEL / 8 / CONFIG_BSP_LCD_SPI_DMA_SIZE_DIV;
  188. ESP_ERROR_CHECK(spi_bus_initialize(SPI3_HOST, &qspi_cfg, SPI_DMA_CH_AUTO));
  189. }
  190. void Initializespd2010Display() {
  191. ESP_LOGI(TAG, "Install panel IO");
  192. const esp_lcd_panel_io_spi_config_t io_config = {
  193. .cs_gpio_num = BSP_LCD_SPI_CS,
  194. .dc_gpio_num = -1,
  195. .spi_mode = 3,
  196. .pclk_hz = DRV_LCD_PIXEL_CLK_HZ,
  197. .trans_queue_depth = 2,
  198. .lcd_cmd_bits = DRV_LCD_CMD_BITS,
  199. .lcd_param_bits = DRV_LCD_PARAM_BITS,
  200. .flags = {
  201. .quad_mode = true,
  202. },
  203. };
  204. spd2010_vendor_config_t vendor_config = {
  205. .flags = {
  206. .use_qspi_interface = 1,
  207. },
  208. };
  209. esp_lcd_new_panel_io_spi((esp_lcd_spi_bus_handle_t)BSP_LCD_SPI_NUM, &io_config, &panel_io_);
  210. ESP_LOGD(TAG, "Install LCD driver");
  211. const esp_lcd_panel_dev_config_t panel_config = {
  212. .reset_gpio_num = BSP_LCD_GPIO_RST, // Shared with Touch reset
  213. .rgb_ele_order = DRV_LCD_RGB_ELEMENT_ORDER,
  214. .bits_per_pixel = DRV_LCD_BITS_PER_PIXEL,
  215. .vendor_config = &vendor_config,
  216. };
  217. esp_lcd_new_panel_spd2010(panel_io_, &panel_config, &panel_);
  218. esp_lcd_panel_reset(panel_);
  219. esp_lcd_panel_init(panel_);
  220. esp_lcd_panel_mirror(panel_, DISPLAY_MIRROR_X, DISPLAY_MIRROR_Y);
  221. esp_lcd_panel_disp_on_off(panel_, true);
  222. display_ = new SpiLcdDisplay(panel_io_, panel_,
  223. DISPLAY_WIDTH, DISPLAY_HEIGHT, DISPLAY_OFFSET_X, DISPLAY_OFFSET_Y, DISPLAY_MIRROR_X, DISPLAY_MIRROR_Y, DISPLAY_SWAP_XY,
  224. {
  225. .text_font = &font_puhui_30_4,
  226. .icon_font = &font_awesome_30_4,
  227. .emoji_font = font_emoji_64_init(),
  228. });
  229. // 使每次刷新的起始列数索引是4的倍数且列数总数是4的倍数,以满足SPD2010的要求
  230. lv_display_add_event_cb(lv_display_get_default(), [](lv_event_t *e) {
  231. lv_area_t *area = (lv_area_t *)lv_event_get_param(e);
  232. uint16_t x1 = area->x1;
  233. uint16_t x2 = area->x2;
  234. // round the start of area down to the nearest 4N number
  235. area->x1 = (x1 >> 2) << 2;
  236. // round the end of area up to the nearest 4M+3 number
  237. area->x2 = ((x2 >> 2) << 2) + 3;
  238. }, LV_EVENT_INVALIDATE_AREA, NULL);
  239. }
  240. // 物联网初始化,添加对 AI 可见设备
  241. void InitializeIot() {
  242. auto& thing_manager = iot::ThingManager::GetInstance();
  243. thing_manager.AddThing(iot::CreateThing("Speaker"));
  244. thing_manager.AddThing(iot::CreateThing("Screen"));
  245. }
  246. public:
  247. SensecapWatcher(){
  248. ESP_LOGI(TAG, "Initialize Sensecap Watcher");
  249. InitializePowerSaveTimer();
  250. InitializeI2c();
  251. InitializeSpi();
  252. InitializeExpander();
  253. InitializeButton();
  254. InitializeKnob();
  255. Initializespd2010Display();
  256. InitializeIot();
  257. GetBacklight()->RestoreBrightness();
  258. }
  259. virtual AudioCodec* GetAudioCodec() override {
  260. static SensecapAudioCodec audio_codec(
  261. i2c_bus_,
  262. AUDIO_INPUT_SAMPLE_RATE,
  263. AUDIO_OUTPUT_SAMPLE_RATE,
  264. AUDIO_I2S_GPIO_MCLK,
  265. AUDIO_I2S_GPIO_BCLK,
  266. AUDIO_I2S_GPIO_WS,
  267. AUDIO_I2S_GPIO_DOUT,
  268. AUDIO_I2S_GPIO_DIN,
  269. AUDIO_CODEC_PA_PIN,
  270. AUDIO_CODEC_ES8311_ADDR,
  271. AUDIO_CODEC_ES7243E_ADDR,
  272. AUDIO_INPUT_REFERENCE);
  273. return &audio_codec;
  274. }
  275. virtual Display* GetDisplay() override {
  276. return display_;
  277. }
  278. virtual Backlight* GetBacklight() override {
  279. static PwmBacklight backlight(DISPLAY_BACKLIGHT_PIN, DISPLAY_BACKLIGHT_OUTPUT_INVERT);
  280. return &backlight;
  281. }
  282. // 根据 https://github.com/Seeed-Studio/OSHW-SenseCAP-Watcher/blob/main/Hardware/SenseCAP_Watcher_v1.0_SCH.pdf
  283. // RGB LED型号为 ws2813 mini, 连接在GPIO 40,供电电压 3.3v, 没有连接 BIN 双信号线
  284. // 可以直接兼容SingleLED采用的ws2812
  285. virtual Led* GetLed() override {
  286. static SingleLed led(BUILTIN_LED_GPIO);
  287. return &led;
  288. }
  289. virtual void SetPowerSaveMode(bool enabled) override {
  290. if (!enabled) {
  291. power_save_timer_->WakeUp();
  292. }
  293. WifiBoard::SetPowerSaveMode(enabled);
  294. }
  295. };
  296. DECLARE_BOARD(SensecapWatcher);