sensecap_watcher.cc 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612
  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 "knob.h"
  9. #include "config.h"
  10. #include "led/single_led.h"
  11. #include "power_save_timer.h"
  12. #include "sscma_camera.h"
  13. #include <esp_log.h>
  14. #include "esp_check.h"
  15. #include <esp_lcd_panel_io.h>
  16. #include <esp_lcd_panel_ops.h>
  17. #include <esp_lcd_spd2010.h>
  18. #include <esp_adc/adc_oneshot.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. #include <esp_console.h>
  28. #include <esp_mac.h>
  29. #include <nvs_flash.h>
  30. #include "assets/lang_config.h"
  31. #define TAG "sensecap_watcher"
  32. LV_FONT_DECLARE(font_puhui_30_4);
  33. LV_FONT_DECLARE(font_awesome_20_4);
  34. class CustomLcdDisplay : public SpiLcdDisplay {
  35. public:
  36. CustomLcdDisplay(esp_lcd_panel_io_handle_t io_handle,
  37. esp_lcd_panel_handle_t panel_handle,
  38. int width,
  39. int height,
  40. int offset_x,
  41. int offset_y,
  42. bool mirror_x,
  43. bool mirror_y,
  44. bool swap_xy)
  45. : SpiLcdDisplay(io_handle, panel_handle, width, height, offset_x, offset_y, mirror_x, mirror_y, swap_xy,
  46. {
  47. .text_font = &font_puhui_30_4,
  48. .icon_font = &font_awesome_20_4,
  49. .emoji_font = font_emoji_64_init(),
  50. }) {
  51. DisplayLockGuard lock(this);
  52. lv_obj_set_size(status_bar_, LV_HOR_RES, fonts_.text_font->line_height * 2 + 10);
  53. lv_obj_set_style_layout(status_bar_, LV_LAYOUT_NONE, 0);
  54. lv_obj_set_style_pad_top(status_bar_, 10, 0);
  55. lv_obj_set_style_pad_bottom(status_bar_, 1, 0);
  56. // 针对圆形屏幕调整位置
  57. // network battery mute //
  58. // status //
  59. lv_obj_align(battery_label_, LV_ALIGN_TOP_MID, -2.5*fonts_.icon_font->line_height, 0);
  60. lv_obj_align(network_label_, LV_ALIGN_TOP_MID, -0.5*fonts_.icon_font->line_height, 0);
  61. lv_obj_align(mute_label_, LV_ALIGN_TOP_MID, 1.5*fonts_.icon_font->line_height, 0);
  62. lv_obj_align(status_label_, LV_ALIGN_BOTTOM_MID, 0, 0);
  63. lv_obj_set_flex_grow(status_label_, 0);
  64. lv_obj_set_width(status_label_, LV_HOR_RES * 0.75);
  65. lv_label_set_long_mode(status_label_, LV_LABEL_LONG_SCROLL_CIRCULAR);
  66. lv_obj_align(notification_label_, LV_ALIGN_BOTTOM_MID, 0, 0);
  67. lv_obj_set_width(notification_label_, LV_HOR_RES * 0.75);
  68. lv_label_set_long_mode(notification_label_, LV_LABEL_LONG_SCROLL_CIRCULAR);
  69. lv_obj_align(low_battery_popup_, LV_ALIGN_BOTTOM_MID, 0, -20);
  70. lv_obj_set_style_bg_color(low_battery_popup_, lv_color_hex(0xFF0000), 0);
  71. lv_obj_set_width(low_battery_label_, LV_HOR_RES * 0.75);
  72. lv_label_set_long_mode(low_battery_label_, LV_LABEL_LONG_SCROLL_CIRCULAR);
  73. }
  74. };
  75. class SensecapWatcher : public WifiBoard {
  76. private:
  77. i2c_master_bus_handle_t i2c_bus_;
  78. LcdDisplay* display_;
  79. std::unique_ptr<Knob> knob_;
  80. esp_io_expander_handle_t io_exp_handle;
  81. button_handle_t btns;
  82. PowerSaveTimer* power_save_timer_;
  83. esp_lcd_panel_io_handle_t panel_io_ = nullptr;
  84. esp_lcd_panel_handle_t panel_ = nullptr;
  85. uint32_t long_press_cnt_;
  86. button_driver_t* btn_driver_ = nullptr;
  87. static SensecapWatcher* instance_;
  88. SscmaCamera* camera_ = nullptr;
  89. void InitializePowerSaveTimer() {
  90. power_save_timer_ = new PowerSaveTimer(-1, 60, 300);
  91. power_save_timer_->OnEnterSleepMode([this]() {
  92. ESP_LOGI(TAG, "Enabling sleep mode");
  93. auto display = GetDisplay();
  94. display->SetChatMessage("system", "");
  95. display->SetEmotion("sleepy");
  96. GetBacklight()->SetBrightness(10);
  97. });
  98. power_save_timer_->OnExitSleepMode([this]() {
  99. auto display = GetDisplay();
  100. display->SetChatMessage("system", "");
  101. display->SetEmotion("neutral");
  102. GetBacklight()->RestoreBrightness();
  103. });
  104. power_save_timer_->OnShutdownRequest([this]() {
  105. ESP_LOGI(TAG, "Shutting down");
  106. bool is_charging = (IoExpanderGetLevel(BSP_PWR_VBUS_IN_DET) == 0);
  107. if (is_charging) {
  108. ESP_LOGI(TAG, "charging");
  109. GetBacklight()->SetBrightness(0);
  110. } else {
  111. IoExpanderSetLevel(BSP_PWR_SYSTEM, 0);
  112. }
  113. });
  114. power_save_timer_->SetEnabled(true);
  115. }
  116. void InitializeI2c() {
  117. // Initialize I2C peripheral
  118. i2c_master_bus_config_t i2c_bus_cfg = {
  119. .i2c_port = (i2c_port_t)0,
  120. .sda_io_num = BSP_GENERAL_I2C_SDA,
  121. .scl_io_num = BSP_GENERAL_I2C_SCL,
  122. .clk_source = I2C_CLK_SRC_DEFAULT,
  123. .glitch_ignore_cnt = 7,
  124. .intr_priority = 0,
  125. .trans_queue_depth = 0,
  126. .flags = {
  127. .enable_internal_pullup = 1,
  128. },
  129. };
  130. ESP_ERROR_CHECK(i2c_new_master_bus(&i2c_bus_cfg, &i2c_bus_));
  131. // pulldown for lcd i2c
  132. const gpio_config_t io_config = {
  133. .pin_bit_mask = (1ULL << BSP_TOUCH_I2C_SDA) | (1ULL << BSP_TOUCH_I2C_SCL) | (1ULL << BSP_SPI3_HOST_PCLK) | (1ULL << BSP_SPI3_HOST_DATA0) | (1ULL << BSP_SPI3_HOST_DATA1)
  134. | (1ULL << BSP_SPI3_HOST_DATA2) | (1ULL << BSP_SPI3_HOST_DATA3) | (1ULL << BSP_LCD_SPI_CS) | (1UL << DISPLAY_BACKLIGHT_PIN),
  135. .mode = GPIO_MODE_OUTPUT,
  136. .pull_up_en = GPIO_PULLUP_DISABLE,
  137. .pull_down_en = GPIO_PULLDOWN_DISABLE,
  138. .intr_type = GPIO_INTR_DISABLE,
  139. };
  140. gpio_config(&io_config);
  141. gpio_set_level(BSP_TOUCH_I2C_SDA, 0);
  142. gpio_set_level(BSP_TOUCH_I2C_SCL, 0);
  143. gpio_set_level(BSP_LCD_SPI_CS, 0);
  144. gpio_set_level(DISPLAY_BACKLIGHT_PIN, 0);
  145. gpio_set_level(BSP_SPI3_HOST_PCLK, 0);
  146. gpio_set_level(BSP_SPI3_HOST_DATA0, 0);
  147. gpio_set_level(BSP_SPI3_HOST_DATA1, 0);
  148. gpio_set_level(BSP_SPI3_HOST_DATA2, 0);
  149. gpio_set_level(BSP_SPI3_HOST_DATA3, 0);
  150. }
  151. esp_err_t IoExpanderSetLevel(uint16_t pin_mask, uint8_t level) {
  152. return esp_io_expander_set_level(io_exp_handle, pin_mask, level);
  153. }
  154. uint8_t IoExpanderGetLevel(uint16_t pin_mask) {
  155. uint32_t pin_val = 0;
  156. esp_io_expander_get_level(io_exp_handle, DRV_IO_EXP_INPUT_MASK, &pin_val);
  157. pin_mask &= DRV_IO_EXP_INPUT_MASK;
  158. return (uint8_t)((pin_val & pin_mask) ? 1 : 0);
  159. }
  160. void InitializeExpander() {
  161. esp_err_t ret = ESP_OK;
  162. esp_io_expander_new_i2c_tca95xx_16bit(i2c_bus_, ESP_IO_EXPANDER_I2C_TCA9555_ADDRESS_001, &io_exp_handle);
  163. ret |= esp_io_expander_set_dir(io_exp_handle, DRV_IO_EXP_INPUT_MASK, IO_EXPANDER_INPUT);
  164. ret |= esp_io_expander_set_dir(io_exp_handle, DRV_IO_EXP_OUTPUT_MASK, IO_EXPANDER_OUTPUT);
  165. ret |= esp_io_expander_set_level(io_exp_handle, DRV_IO_EXP_OUTPUT_MASK, 0);
  166. ret |= esp_io_expander_set_level(io_exp_handle, BSP_PWR_SYSTEM, 1);
  167. vTaskDelay(100 / portTICK_PERIOD_MS);
  168. ret |= esp_io_expander_set_level(io_exp_handle, BSP_PWR_START_UP, 1);
  169. vTaskDelay(50 / portTICK_PERIOD_MS);
  170. uint32_t pin_val = 0;
  171. ret |= esp_io_expander_get_level(io_exp_handle, DRV_IO_EXP_INPUT_MASK, &pin_val);
  172. ESP_LOGI(TAG, "IO expander initialized: %x", DRV_IO_EXP_OUTPUT_MASK | (uint16_t)pin_val);
  173. assert(ret == ESP_OK);
  174. }
  175. void OnKnobRotate(bool clockwise) {
  176. auto codec = GetAudioCodec();
  177. int current_volume = codec->output_volume();
  178. int new_volume = current_volume + (clockwise ? -5 : 5);
  179. // 确保音量在有效范围内
  180. if (new_volume > 100) {
  181. new_volume = 100;
  182. ESP_LOGW(TAG, "Volume reached maximum limit: %d", new_volume);
  183. } else if (new_volume < 0) {
  184. new_volume = 0;
  185. ESP_LOGW(TAG, "Volume reached minimum limit: %d", new_volume);
  186. }
  187. codec->SetOutputVolume(new_volume);
  188. ESP_LOGI(TAG, "Volume changed from %d to %d", current_volume, new_volume);
  189. // 显示通知前检查实际变化
  190. if (new_volume != codec->output_volume()) {
  191. ESP_LOGE(TAG, "Failed to set volume! Expected:%d Actual:%d",
  192. new_volume, codec->output_volume());
  193. }
  194. GetDisplay()->ShowNotification(std::string(Lang::Strings::VOLUME) + ": "+std::to_string(codec->output_volume()));
  195. power_save_timer_->WakeUp();
  196. }
  197. void InitializeKnob() {
  198. knob_ = std::make_unique<Knob>(BSP_KNOB_A_PIN, BSP_KNOB_B_PIN);
  199. knob_->OnRotate([this](bool clockwise) {
  200. ESP_LOGD(TAG, "Knob rotation detected. Clockwise:%s", clockwise ? "true" : "false");
  201. OnKnobRotate(clockwise);
  202. });
  203. ESP_LOGI(TAG, "Knob initialized with pins A:%d B:%d", BSP_KNOB_A_PIN, BSP_KNOB_B_PIN);
  204. }
  205. void InitializeButton() {
  206. // 设置静态实例指针
  207. instance_ = this;
  208. // watcher 是通过长按滚轮进行开机的, 需要等待滚轮释放, 否则用户开机松手时可能会误触成单击
  209. ESP_LOGI(TAG, "waiting for knob button release");
  210. while(IoExpanderGetLevel(BSP_KNOB_BTN) == 0) {
  211. vTaskDelay(pdMS_TO_TICKS(50));
  212. }
  213. button_config_t btn_config = {
  214. .long_press_time = 2000,
  215. .short_press_time = 0
  216. };
  217. btn_driver_ = (button_driver_t*)calloc(1, sizeof(button_driver_t));
  218. btn_driver_->enable_power_save = false;
  219. btn_driver_->get_key_level = [](button_driver_t *button_driver) -> uint8_t {
  220. return !instance_->IoExpanderGetLevel(BSP_KNOB_BTN);
  221. };
  222. ESP_ERROR_CHECK(iot_button_create(&btn_config, btn_driver_, &btns));
  223. iot_button_register_cb(btns, BUTTON_SINGLE_CLICK, nullptr, [](void* button_handle, void* usr_data) {
  224. auto self = static_cast<SensecapWatcher*>(usr_data);
  225. auto& app = Application::GetInstance();
  226. if (app.GetDeviceState() == kDeviceStateStarting && !WifiStation::GetInstance().IsConnected()) {
  227. self->ResetWifiConfiguration();
  228. }
  229. self->power_save_timer_->WakeUp();
  230. app.ToggleChatState();
  231. }, this);
  232. iot_button_register_cb(btns, BUTTON_LONG_PRESS_START, nullptr, [](void* button_handle, void* usr_data) {
  233. auto self = static_cast<SensecapWatcher*>(usr_data);
  234. bool is_charging = (self->IoExpanderGetLevel(BSP_PWR_VBUS_IN_DET) == 0);
  235. self->long_press_cnt_ = 0;
  236. if (is_charging) {
  237. ESP_LOGI(TAG, "charging");
  238. } else {
  239. self->IoExpanderSetLevel(BSP_PWR_LCD, 0);
  240. self->IoExpanderSetLevel(BSP_PWR_SYSTEM, 0);
  241. }
  242. }, this);
  243. iot_button_register_cb(btns, BUTTON_LONG_PRESS_HOLD, nullptr, [](void* button_handle, void* usr_data) {
  244. auto self = static_cast<SensecapWatcher*>(usr_data);
  245. self->long_press_cnt_++; // 每隔20ms加一
  246. // 长按10s 恢复出厂设置: 2+0.02*400 = 10
  247. if (self->long_press_cnt_ > 400) {
  248. ESP_LOGI(TAG, "Factory reset");
  249. nvs_flash_erase();
  250. esp_restart();
  251. }
  252. }, this);
  253. }
  254. void InitializeSpi() {
  255. ESP_LOGI(TAG, "Initialize SSCMA SPI bus");
  256. spi_bus_config_t spi_cfg = {0};
  257. spi_cfg.mosi_io_num = BSP_SPI2_HOST_MOSI;
  258. spi_cfg.miso_io_num = BSP_SPI2_HOST_MISO;
  259. spi_cfg.sclk_io_num = BSP_SPI2_HOST_SCLK;
  260. spi_cfg.quadwp_io_num = -1;
  261. spi_cfg.quadhd_io_num = -1;
  262. spi_cfg.isr_cpu_id = ESP_INTR_CPU_AFFINITY_1;
  263. spi_cfg.max_transfer_sz = 4095;
  264. ESP_ERROR_CHECK(spi_bus_initialize(SPI2_HOST, &spi_cfg, SPI_DMA_CH_AUTO));
  265. ESP_LOGI(TAG, "Initialize QSPI bus");
  266. spi_bus_config_t qspi_cfg = {0};
  267. qspi_cfg.sclk_io_num = BSP_SPI3_HOST_PCLK;
  268. qspi_cfg.data0_io_num = BSP_SPI3_HOST_DATA0;
  269. qspi_cfg.data1_io_num = BSP_SPI3_HOST_DATA1;
  270. qspi_cfg.data2_io_num = BSP_SPI3_HOST_DATA2;
  271. qspi_cfg.data3_io_num = BSP_SPI3_HOST_DATA3;
  272. qspi_cfg.max_transfer_sz = DISPLAY_WIDTH * DISPLAY_HEIGHT * DRV_LCD_BITS_PER_PIXEL / 8 / CONFIG_BSP_LCD_SPI_DMA_SIZE_DIV;
  273. ESP_ERROR_CHECK(spi_bus_initialize(SPI3_HOST, &qspi_cfg, SPI_DMA_CH_AUTO));
  274. }
  275. void Initializespd2010Display() {
  276. ESP_LOGI(TAG, "Install panel IO");
  277. const esp_lcd_panel_io_spi_config_t io_config = {
  278. .cs_gpio_num = BSP_LCD_SPI_CS,
  279. .dc_gpio_num = -1,
  280. .spi_mode = 3,
  281. .pclk_hz = DRV_LCD_PIXEL_CLK_HZ,
  282. .trans_queue_depth = 2,
  283. .lcd_cmd_bits = DRV_LCD_CMD_BITS,
  284. .lcd_param_bits = DRV_LCD_PARAM_BITS,
  285. .flags = {
  286. .quad_mode = true,
  287. },
  288. };
  289. spd2010_vendor_config_t vendor_config = {
  290. .flags = {
  291. .use_qspi_interface = 1,
  292. },
  293. };
  294. esp_lcd_new_panel_io_spi((esp_lcd_spi_bus_handle_t)BSP_LCD_SPI_NUM, &io_config, &panel_io_);
  295. ESP_LOGD(TAG, "Install LCD driver");
  296. const esp_lcd_panel_dev_config_t panel_config = {
  297. .reset_gpio_num = BSP_LCD_GPIO_RST, // Shared with Touch reset
  298. .rgb_ele_order = DRV_LCD_RGB_ELEMENT_ORDER,
  299. .bits_per_pixel = DRV_LCD_BITS_PER_PIXEL,
  300. .vendor_config = &vendor_config,
  301. };
  302. esp_lcd_new_panel_spd2010(panel_io_, &panel_config, &panel_);
  303. esp_lcd_panel_reset(panel_);
  304. esp_lcd_panel_init(panel_);
  305. esp_lcd_panel_mirror(panel_, DISPLAY_MIRROR_X, DISPLAY_MIRROR_Y);
  306. esp_lcd_panel_disp_on_off(panel_, true);
  307. display_ = new CustomLcdDisplay(panel_io_, panel_,
  308. DISPLAY_WIDTH, DISPLAY_HEIGHT, DISPLAY_OFFSET_X, DISPLAY_OFFSET_Y, DISPLAY_MIRROR_X, DISPLAY_MIRROR_Y, DISPLAY_SWAP_XY);
  309. // 使每次刷新的起始列数索引是4的倍数且列数总数是4的倍数,以满足SPD2010的要求
  310. lv_display_add_event_cb(lv_display_get_default(), [](lv_event_t *e) {
  311. lv_area_t *area = (lv_area_t *)lv_event_get_param(e);
  312. uint16_t x1 = area->x1;
  313. uint16_t x2 = area->x2;
  314. // round the start of area down to the nearest 4N number
  315. area->x1 = (x1 >> 2) << 2;
  316. // round the end of area up to the nearest 4M+3 number
  317. area->x2 = ((x2 >> 2) << 2) + 3;
  318. }, LV_EVENT_INVALIDATE_AREA, NULL);
  319. }
  320. uint16_t BatterygetVoltage(void) {
  321. static bool initialized = false;
  322. static adc_oneshot_unit_handle_t adc_handle;
  323. static adc_cali_handle_t cali_handle = NULL;
  324. if (!initialized) {
  325. adc_oneshot_unit_init_cfg_t init_config = {
  326. .unit_id = ADC_UNIT_1,
  327. };
  328. adc_oneshot_new_unit(&init_config, &adc_handle);
  329. adc_oneshot_chan_cfg_t ch_config = {
  330. .atten = BSP_BAT_ADC_ATTEN,
  331. .bitwidth = ADC_BITWIDTH_DEFAULT,
  332. };
  333. adc_oneshot_config_channel(adc_handle, BSP_BAT_ADC_CHAN, &ch_config);
  334. adc_cali_curve_fitting_config_t cali_config = {
  335. .unit_id = ADC_UNIT_1,
  336. .chan = BSP_BAT_ADC_CHAN,
  337. .atten = BSP_BAT_ADC_ATTEN,
  338. .bitwidth = ADC_BITWIDTH_DEFAULT,
  339. };
  340. if (adc_cali_create_scheme_curve_fitting(&cali_config, &cali_handle) == ESP_OK) {
  341. initialized = true;
  342. }
  343. }
  344. if (initialized) {
  345. int raw_value = 0;
  346. int voltage = 0; // mV
  347. adc_oneshot_read(adc_handle, BSP_BAT_ADC_CHAN, &raw_value);
  348. adc_cali_raw_to_voltage(cali_handle, raw_value, &voltage);
  349. voltage = voltage * 82 / 20;
  350. // ESP_LOGI(TAG, "voltage: %dmV", voltage);
  351. return (uint16_t)voltage;
  352. }
  353. return 0;
  354. }
  355. uint8_t BatterygetPercent(bool print = false) {
  356. int voltage = 0;
  357. for (uint8_t i = 0; i < 10; i++) {
  358. voltage += BatterygetVoltage();
  359. }
  360. voltage /= 10;
  361. int percent = (-1 * voltage * voltage + 9016 * voltage - 19189000) / 10000;
  362. percent = (percent > 100) ? 100 : (percent < 0) ? 0 : percent;
  363. if (print) {
  364. printf("voltage: %dmV, percentage: %d%%\r\n", voltage, percent);
  365. }
  366. return (uint8_t)percent;
  367. }
  368. void InitializeCmd() {
  369. esp_console_repl_t *repl = NULL;
  370. esp_console_repl_config_t repl_config = ESP_CONSOLE_REPL_CONFIG_DEFAULT();
  371. repl_config.max_cmdline_length = 1024;
  372. repl_config.prompt = "SenseCAP>";
  373. const esp_console_cmd_t cmd1 = {
  374. .command = "reboot",
  375. .help = "reboot the device",
  376. .hint = nullptr,
  377. .func = [](int argc, char** argv) -> int {
  378. esp_restart();
  379. return 0;
  380. },
  381. .argtable = nullptr
  382. };
  383. ESP_ERROR_CHECK(esp_console_cmd_register(&cmd1));
  384. const esp_console_cmd_t cmd2 = {
  385. .command = "shutdown",
  386. .help = "shutdown the device",
  387. .hint = nullptr,
  388. .func = NULL,
  389. .argtable = NULL,
  390. .func_w_context = [](void *context,int argc, char** argv) -> int {
  391. auto self = static_cast<SensecapWatcher*>(context);
  392. self->GetBacklight()->SetBrightness(0);
  393. self->IoExpanderSetLevel(BSP_PWR_SYSTEM, 0);
  394. return 0;
  395. },
  396. .context =this
  397. };
  398. ESP_ERROR_CHECK(esp_console_cmd_register(&cmd2));
  399. const esp_console_cmd_t cmd3 = {
  400. .command = "battery",
  401. .help = "get battery percent",
  402. .hint = NULL,
  403. .func = NULL,
  404. .argtable = NULL,
  405. .func_w_context = [](void *context,int argc, char** argv) -> int {
  406. auto self = static_cast<SensecapWatcher*>(context);
  407. self->BatterygetPercent(true);
  408. return 0;
  409. },
  410. .context =this
  411. };
  412. ESP_ERROR_CHECK(esp_console_cmd_register(&cmd3));
  413. const esp_console_cmd_t cmd4 = {
  414. .command = "factory_reset",
  415. .help = "factory reset and reboot the device",
  416. .hint = NULL,
  417. .func = NULL,
  418. .argtable = NULL,
  419. .func_w_context = [](void *context,int argc, char** argv) -> int {
  420. nvs_flash_erase();
  421. esp_restart();
  422. return 0;
  423. },
  424. .context =this
  425. };
  426. ESP_ERROR_CHECK(esp_console_cmd_register(&cmd4));
  427. const esp_console_cmd_t cmd5 = {
  428. .command = "read_mac",
  429. .help = "Read mac address",
  430. .hint = NULL,
  431. .func = NULL,
  432. .argtable = NULL,
  433. .func_w_context = [](void *context,int argc, char** argv) -> int {
  434. uint8_t mac[6];
  435. esp_read_mac(mac, ESP_MAC_WIFI_STA);
  436. printf("wifi_sta_mac: " MACSTR "\n", MAC2STR(mac));
  437. esp_read_mac(mac, ESP_MAC_WIFI_SOFTAP);
  438. printf("wifi_softap_mac: " MACSTR "\n", MAC2STR(mac));
  439. esp_read_mac(mac, ESP_MAC_BT);
  440. printf("bt_mac: " MACSTR "\n", MAC2STR(mac));
  441. return 0;
  442. },
  443. .context =this
  444. };
  445. ESP_ERROR_CHECK(esp_console_cmd_register(&cmd5));
  446. esp_console_dev_uart_config_t hw_config = ESP_CONSOLE_DEV_UART_CONFIG_DEFAULT();
  447. ESP_ERROR_CHECK(esp_console_new_repl_uart(&hw_config, &repl_config, &repl));
  448. ESP_ERROR_CHECK(esp_console_start_repl(repl));
  449. }
  450. void InitializeCamera() {
  451. ESP_LOGI(TAG, "Initialize Camera");
  452. // !!!NOTE: SD Card use same SPI bus as sscma client, so we need to disable SD card CS pin first
  453. const gpio_config_t io_config = {
  454. .pin_bit_mask = (1ULL << BSP_SD_SPI_CS),
  455. .mode = GPIO_MODE_OUTPUT,
  456. .pull_up_en = GPIO_PULLUP_ENABLE,
  457. .pull_down_en = GPIO_PULLDOWN_DISABLE,
  458. .intr_type = GPIO_INTR_DISABLE,
  459. };
  460. esp_err_t ret = gpio_config(&io_config);
  461. if (ret != ESP_OK)
  462. return;
  463. gpio_set_level(BSP_SD_SPI_CS, 1);
  464. camera_ = new SscmaCamera(io_exp_handle);
  465. }
  466. public:
  467. SensecapWatcher() {
  468. ESP_LOGI(TAG, "Initialize Sensecap Watcher");
  469. InitializePowerSaveTimer();
  470. InitializeI2c();
  471. InitializeSpi();
  472. InitializeExpander();
  473. InitializeCmd(); //工厂生产测试使用
  474. InitializeButton();
  475. InitializeKnob();
  476. Initializespd2010Display();
  477. GetBacklight()->RestoreBrightness(); // 对于不带摄像头的版本,InitializeCamera需要3s, 所以先恢复背光亮度
  478. InitializeCamera();
  479. }
  480. virtual AudioCodec* GetAudioCodec() override {
  481. static SensecapAudioCodec audio_codec(
  482. i2c_bus_,
  483. AUDIO_INPUT_SAMPLE_RATE,
  484. AUDIO_OUTPUT_SAMPLE_RATE,
  485. AUDIO_I2S_GPIO_MCLK,
  486. AUDIO_I2S_GPIO_BCLK,
  487. AUDIO_I2S_GPIO_WS,
  488. AUDIO_I2S_GPIO_DOUT,
  489. AUDIO_I2S_GPIO_DIN,
  490. AUDIO_CODEC_PA_PIN,
  491. AUDIO_CODEC_ES8311_ADDR,
  492. AUDIO_CODEC_ES7243E_ADDR,
  493. AUDIO_INPUT_REFERENCE);
  494. return &audio_codec;
  495. }
  496. virtual Display* GetDisplay() override {
  497. return display_;
  498. }
  499. virtual Backlight* GetBacklight() override {
  500. static PwmBacklight backlight(DISPLAY_BACKLIGHT_PIN, DISPLAY_BACKLIGHT_OUTPUT_INVERT);
  501. return &backlight;
  502. }
  503. // 根据 https://github.com/Seeed-Studio/OSHW-SenseCAP-Watcher/blob/main/Hardware/SenseCAP_Watcher_v1.0_SCH.pdf
  504. // RGB LED型号为 ws2813 mini, 连接在GPIO 40,供电电压 3.3v, 没有连接 BIN 双信号线
  505. // 可以直接兼容SingleLED采用的ws2812
  506. virtual Led* GetLed() override {
  507. static SingleLed led(BUILTIN_LED_GPIO);
  508. return &led;
  509. }
  510. virtual void SetPowerSaveMode(bool enabled) override {
  511. if (!enabled) {
  512. power_save_timer_->WakeUp();
  513. }
  514. WifiBoard::SetPowerSaveMode(enabled);
  515. }
  516. virtual bool GetBatteryLevel(int &level, bool& charging, bool& discharging) override {
  517. static bool last_discharging = false;
  518. charging = (IoExpanderGetLevel(BSP_PWR_VBUS_IN_DET) == 0);
  519. discharging = !charging;
  520. level = (int)BatterygetPercent(false);
  521. if (discharging != last_discharging) {
  522. power_save_timer_->SetEnabled(discharging);
  523. last_discharging = discharging;
  524. }
  525. if (level <= 1 && discharging) {
  526. ESP_LOGI(TAG, "Battery level is low, shutting down");
  527. IoExpanderSetLevel(BSP_PWR_SYSTEM, 0);
  528. }
  529. return true;
  530. }
  531. virtual Camera* GetCamera() override {
  532. return camera_;
  533. }
  534. };
  535. DECLARE_BOARD(SensecapWatcher);
  536. // 定义静态成员变量
  537. SensecapWatcher* SensecapWatcher::instance_ = nullptr;