forked from xiaozhi/xiaozhi-esp32
[优化]-jiuchuan-s3-优化电源管理架构 (#875)
* 添加 jiuchuang-s3 开发板支持 * 增加编译指南 * 优开发板支持包文件目录,更新README.md 删除了多余板载文件 [* ]README.md -> 更新了编译指南和烧录指南 * 修改板级支持包名 * 使用乐鑫提供的电源监测 * 修复部分代码格式问题 * 解决合并冲突 * 解决部分合并内容 * 完善合并内容 * 修复电量映射表错误 * 优化电源管理架构-使用rtc-io电源保持 --------- Co-authored-by: unknown <jake12355> Co-authored-by: Hangon66 <2630612613@qq.com>
This commit is contained in:
@@ -132,14 +132,9 @@ private:
|
|||||||
}
|
}
|
||||||
|
|
||||||
void InitializeButtons() {
|
void InitializeButtons() {
|
||||||
auto& powerCtrl = PowerController::Instance();
|
|
||||||
powerCtrl.SetState(PowerController::PowerState::ACTIVE); // 确保初始状态为ACTIVE
|
|
||||||
|
|
||||||
// 配置GPIO
|
// 配置GPIO
|
||||||
ESP_LOGI(TAG, "Configuring power button GPIO");
|
ESP_LOGI(TAG, "Configuring power button GPIO");
|
||||||
GpioManager::Config(GPIO_NUM_3, GpioManager::GpioMode::INPUT_PULLDOWN);
|
GpioManager::Config(GPIO_NUM_3, GpioManager::GpioMode::INPUT_PULLDOWN);
|
||||||
GpioManager::Config(PWR_EN_GPIO, GpioManager::GpioMode::OUTPUT);
|
|
||||||
GpioManager::SetLevel(PWR_EN_GPIO, 1); // 确保电源使能
|
|
||||||
|
|
||||||
boot_button_.OnClick([this]() {
|
boot_button_.OnClick([this]() {
|
||||||
ESP_LOGI(TAG, "Boot button clicked");
|
ESP_LOGI(TAG, "Boot button clicked");
|
||||||
@@ -150,7 +145,7 @@ private:
|
|||||||
ESP_LOGI(TAG, "Power button initial state: %d", GpioManager::GetLevel(PWR_BUTTON_GPIO));
|
ESP_LOGI(TAG, "Power button initial state: %d", GpioManager::GetLevel(PWR_BUTTON_GPIO));
|
||||||
|
|
||||||
// 高电平有效长按关机逻辑
|
// 高电平有效长按关机逻辑
|
||||||
pwr_button_.OnLongPress([&powerCtrl]() {
|
pwr_button_.OnLongPress([this]() {
|
||||||
ESP_LOGI(TAG, "Power button long press detected (high-active)");
|
ESP_LOGI(TAG, "Power button long press detected (high-active)");
|
||||||
|
|
||||||
// 高电平有效防抖确认
|
// 高电平有效防抖确认
|
||||||
@@ -164,26 +159,20 @@ private:
|
|||||||
}
|
}
|
||||||
vTaskDelay(100 / portTICK_PERIOD_MS);
|
vTaskDelay(100 / portTICK_PERIOD_MS);
|
||||||
}
|
}
|
||||||
ESP_LOGI(TAG, "Confirmed power button pressed (level=1)");
|
|
||||||
|
|
||||||
ESP_LOGI(TAG, "Confirmed power button pressed - initiating shutdown");
|
ESP_LOGI(TAG, "Confirmed power button pressed - initiating shutdown");
|
||||||
powerCtrl.SetState(PowerController::PowerState::SHUTDOWN);
|
power_manager_->SetPowerState(PowerState::SHUTDOWN);
|
||||||
|
|
||||||
// 确保状态变更
|
|
||||||
if (powerCtrl.GetState() != PowerController::PowerState::SHUTDOWN) {
|
|
||||||
ESP_LOGE(TAG, "Failed to set shutdown state!");
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
wifi_button.OnClick([this, &powerCtrl]() {
|
wifi_button.OnClick([this]() {
|
||||||
ESP_LOGI(TAG, "Wifi button clicked");
|
ESP_LOGI(TAG, "Wifi button clicked");
|
||||||
power_save_timer_->WakeUp();
|
power_save_timer_->WakeUp();
|
||||||
|
|
||||||
if (powerCtrl.GetState() == PowerController::PowerState::ACTIVE) {
|
|
||||||
ESP_LOGI(TAG, "Resetting WiFi configuration");
|
ESP_LOGI(TAG, "Resetting WiFi configuration");
|
||||||
GpioManager::SetLevel(PWR_EN_GPIO, 1);
|
GpioManager::SetLevel(PWR_EN_GPIO, 1);
|
||||||
ResetWifiConfiguration();
|
ResetWifiConfiguration();
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
cmd_button.OnClick([this]() {
|
cmd_button.OnClick([this]() {
|
||||||
@@ -191,40 +180,6 @@ private:
|
|||||||
power_save_timer_->WakeUp();
|
power_save_timer_->WakeUp();
|
||||||
Application::GetInstance().ToggleChatState();
|
Application::GetInstance().ToggleChatState();
|
||||||
});
|
});
|
||||||
|
|
||||||
// 配置电源状态变更回调(优化版)
|
|
||||||
powerCtrl.OnStateChange([this](PowerController::PowerState newState) {
|
|
||||||
switch(newState) {
|
|
||||||
case PowerController::PowerState::SHUTDOWN: {
|
|
||||||
ESP_LOGI(TAG, "Entering shutdown sequence");
|
|
||||||
|
|
||||||
// 统一唤醒触发条件
|
|
||||||
#ifndef __USER_GPIO_PWRDOWN__
|
|
||||||
ESP_LOGD(TAG, "Configuring high-level wakeup on GPIO%d", PWR_BUTTON_GPIO);
|
|
||||||
ESP_ERROR_CHECK(esp_sleep_enable_ext0_wakeup(PWR_BUTTON_GPIO, 1)); // 高电平唤醒
|
|
||||||
#else
|
|
||||||
ESP_LOGD(TAG, "Powering down via GPIO control");
|
|
||||||
GpioManager::SetLevel(PWR_EN_GPIO, 0);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// 确保所有外设已关闭
|
|
||||||
vTaskDelay(200 / portTICK_PERIOD_MS);
|
|
||||||
ESP_LOGI(TAG, "Initiating deep sleep");
|
|
||||||
|
|
||||||
// 最后状态确认(通过单例访问)
|
|
||||||
if (PowerController::Instance().GetState() != PowerController::PowerState::SHUTDOWN) {
|
|
||||||
ESP_LOGE(TAG, "State inconsistency! Forcing shutdown");
|
|
||||||
}
|
|
||||||
esp_deep_sleep_start();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
default:
|
|
||||||
ESP_LOGD(TAG, "State changed to %d", static_cast<int>(newState));
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void InitializeGC9301isplay() {
|
void InitializeGC9301isplay() {
|
||||||
|
|||||||
@@ -2,17 +2,20 @@
|
|||||||
#include <mutex>
|
#include <mutex>
|
||||||
#include <functional>
|
#include <functional>
|
||||||
#include <driver/gpio.h>
|
#include <driver/gpio.h>
|
||||||
|
#include <driver/rtc_io.h>
|
||||||
#include <esp_log.h>
|
#include <esp_log.h>
|
||||||
|
#include "config.h"
|
||||||
class PowerController {
|
enum class PowerState {
|
||||||
public:
|
|
||||||
enum class PowerState {
|
|
||||||
ACTIVE,
|
ACTIVE,
|
||||||
LIGHT_SLEEP,
|
LIGHT_SLEEP,
|
||||||
DEEP_SLEEP,
|
DEEP_SLEEP,
|
||||||
SHUTDOWN
|
SHUTDOWN
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class PowerController {
|
||||||
|
public:
|
||||||
|
|
||||||
|
|
||||||
static PowerController& Instance() {
|
static PowerController& Instance() {
|
||||||
static PowerController instance;
|
static PowerController instance;
|
||||||
return instance;
|
return instance;
|
||||||
@@ -42,7 +45,11 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
PowerController() = default;
|
PowerController(){
|
||||||
|
rtc_gpio_init(PWR_EN_GPIO);
|
||||||
|
rtc_gpio_set_direction(PWR_EN_GPIO, RTC_GPIO_MODE_OUTPUT_ONLY);
|
||||||
|
rtc_gpio_set_level(PWR_EN_GPIO, 1);
|
||||||
|
}
|
||||||
~PowerController() = default;
|
~PowerController() = default;
|
||||||
|
|
||||||
PowerState currentState_ = PowerState::ACTIVE;
|
PowerState currentState_ = PowerState::ACTIVE;
|
||||||
|
|||||||
@@ -4,8 +4,10 @@
|
|||||||
|
|
||||||
#include <esp_timer.h>
|
#include <esp_timer.h>
|
||||||
#include <driver/gpio.h>
|
#include <driver/gpio.h>
|
||||||
#include <esp_adc/adc_oneshot.h>
|
|
||||||
#include "adc_battery_estimation.h"
|
#include "adc_battery_estimation.h"
|
||||||
|
#include "power_controller.h"
|
||||||
|
#include <driver/rtc_io.h>
|
||||||
|
#include <esp_sleep.h>
|
||||||
|
|
||||||
#define JIUCHUAN_ADC_UNIT (ADC_UNIT_1)
|
#define JIUCHUAN_ADC_UNIT (ADC_UNIT_1)
|
||||||
#define JIUCHUAN_ADC_BITWIDTH (ADC_BITWIDTH_12)
|
#define JIUCHUAN_ADC_BITWIDTH (ADC_BITWIDTH_12)
|
||||||
@@ -14,23 +16,28 @@
|
|||||||
#define JIUCHUAN_RESISTOR_UPPER (200000)
|
#define JIUCHUAN_RESISTOR_UPPER (200000)
|
||||||
#define JIUCHUAN_RESISTOR_LOWER (100000)
|
#define JIUCHUAN_RESISTOR_LOWER (100000)
|
||||||
|
|
||||||
|
#undef TAG
|
||||||
|
#define TAG "PowerManager"
|
||||||
class PowerManager {
|
class PowerManager {
|
||||||
private:
|
private:
|
||||||
esp_timer_handle_t timer_handle_;
|
esp_timer_handle_t timer_handle_;
|
||||||
std::function<void(bool)> on_charging_status_changed_;
|
std::function<void(bool)> on_charging_status_changed_;
|
||||||
std::function<void(bool)> on_low_battery_status_changed_;
|
std::function<void(bool)> on_low_battery_status_changed_;
|
||||||
|
|
||||||
gpio_num_t charging_pin_ = GPIO_NUM_NC;
|
gpio_num_t charging_pin_ = GPIO_NUM_NC;
|
||||||
std::vector<uint16_t> adc_values_;
|
std::vector<uint16_t> adc_values_;
|
||||||
uint32_t battery_level_ = 0;
|
int32_t battery_level_ = 100;
|
||||||
bool is_charging_ = false;
|
bool is_charging_ = false;
|
||||||
bool is_low_battery_ = false;
|
bool is_low_battery_ = false;
|
||||||
|
bool is_empty_battery_ = false;
|
||||||
int ticks_ = 0;
|
int ticks_ = 0;
|
||||||
const int kBatteryAdcInterval = 60;
|
const int kBatteryAdcInterval = 60;
|
||||||
const int kBatteryAdcDataCount = 3;
|
const int kBatteryAdcDataCount = 3;
|
||||||
const int kLowBatteryLevel = 20;
|
const int kLowBatteryLevel = 20;
|
||||||
|
|
||||||
adc_battery_estimation_handle_t adc_battery_estimation_handle;
|
adc_battery_estimation_handle_t adc_battery_estimation_handle;
|
||||||
|
PowerController* power_controller_;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
void CheckBatteryStatus() {
|
void CheckBatteryStatus() {
|
||||||
// Get charging status
|
// Get charging status
|
||||||
@@ -61,11 +68,16 @@ private:
|
|||||||
float battery_capacity_temp = 0;
|
float battery_capacity_temp = 0;
|
||||||
adc_battery_estimation_get_capacity(adc_battery_estimation_handle, &battery_capacity_temp);
|
adc_battery_estimation_get_capacity(adc_battery_estimation_handle, &battery_capacity_temp);
|
||||||
ESP_LOGI("PowerManager", "Battery level: %.1f%%", battery_capacity_temp);
|
ESP_LOGI("PowerManager", "Battery level: %.1f%%", battery_capacity_temp);
|
||||||
battery_level_ = battery_capacity_temp;
|
if(battery_capacity_temp > -10 && battery_capacity_temp <= 0){
|
||||||
|
battery_level_ = 0;
|
||||||
|
}else{
|
||||||
|
battery_level_ = battery_capacity_temp;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public:
|
public:
|
||||||
PowerManager(gpio_num_t pin) : charging_pin_(pin) {
|
PowerManager(gpio_num_t pin) : charging_pin_(pin) {
|
||||||
|
power_controller_ = &PowerController::Instance();
|
||||||
// 初始化充电引脚
|
// 初始化充电引脚
|
||||||
gpio_config_t io_conf = {};
|
gpio_config_t io_conf = {};
|
||||||
io_conf.intr_type = GPIO_INTR_DISABLE;
|
io_conf.intr_type = GPIO_INTR_DISABLE;
|
||||||
@@ -89,14 +101,14 @@ public:
|
|||||||
ESP_ERROR_CHECK(esp_timer_create(&timer_args, &timer_handle_));
|
ESP_ERROR_CHECK(esp_timer_create(&timer_args, &timer_handle_));
|
||||||
ESP_ERROR_CHECK(esp_timer_start_periodic(timer_handle_, 1000000));
|
ESP_ERROR_CHECK(esp_timer_start_periodic(timer_handle_, 1000000));
|
||||||
|
|
||||||
// 初始化 ADC
|
|
||||||
static const battery_point_t battery_ponint_table[]={
|
static const battery_point_t battery_ponint_table[]={
|
||||||
{ 4.2 , 100},
|
{ 4.2 , 100},
|
||||||
{ 4.06 , 80},
|
{ 4.06 , 80},
|
||||||
{ 3.82 , 60},
|
{ 3.82 , 60},
|
||||||
{ 3.58 , 40},
|
{ 3.58 , 40},
|
||||||
{ 3.34 , 20},
|
{ 3.34 , 20},
|
||||||
{ 3.1 , 0}
|
{ 3.1 , 0},
|
||||||
|
{ 3.0 , -10}
|
||||||
};
|
};
|
||||||
|
|
||||||
adc_battery_estimation_t config = {
|
adc_battery_estimation_t config = {
|
||||||
@@ -113,6 +125,8 @@ public:
|
|||||||
};
|
};
|
||||||
|
|
||||||
adc_battery_estimation_handle = adc_battery_estimation_create(&config);
|
adc_battery_estimation_handle = adc_battery_estimation_create(&config);
|
||||||
|
|
||||||
|
RegisterAllCallbacks();
|
||||||
}
|
}
|
||||||
|
|
||||||
~PowerManager() {
|
~PowerManager() {
|
||||||
@@ -120,11 +134,15 @@ public:
|
|||||||
esp_timer_stop(timer_handle_);
|
esp_timer_stop(timer_handle_);
|
||||||
esp_timer_delete(timer_handle_);
|
esp_timer_delete(timer_handle_);
|
||||||
}
|
}
|
||||||
|
if (adc_battery_estimation_handle) {
|
||||||
|
adc_battery_estimation_destroy(adc_battery_estimation_handle);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool IsCharging() {
|
bool IsCharging() {
|
||||||
// 如果电量已经满了,则不再显示充电中
|
// 如果电量已经满了,则不再显示充电中
|
||||||
if (battery_level_ == 100) {
|
if (battery_level_ == 100) {
|
||||||
|
//ESP_LOGI(TAG, "电量已满,不再显示充电中");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return is_charging_;
|
return is_charging_;
|
||||||
@@ -135,10 +153,44 @@ public:
|
|||||||
return !is_charging_;
|
return !is_charging_;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint8_t GetBatteryLevel() {
|
int32_t GetBatteryLevel() {
|
||||||
return battery_level_;
|
return battery_level_;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void RegisterAllCallbacks() {
|
||||||
|
//注册电源状态变更回调函数(优化版)
|
||||||
|
power_controller_->OnStateChange([this](PowerState newState) {
|
||||||
|
switch(newState) {
|
||||||
|
case PowerState::SHUTDOWN: {
|
||||||
|
|
||||||
|
ESP_LOGD(TAG, "关机");
|
||||||
|
|
||||||
|
//取消 PWR_EN 使能
|
||||||
|
/* 防止关机后误唤醒 */
|
||||||
|
ESP_ERROR_CHECK(esp_sleep_enable_ext0_wakeup(PWR_BUTTON_GPIO, 0));
|
||||||
|
ESP_ERROR_CHECK(rtc_gpio_pulldown_en(PWR_BUTTON_GPIO)); // 内部下拉
|
||||||
|
ESP_ERROR_CHECK(rtc_gpio_pullup_dis(PWR_BUTTON_GPIO));
|
||||||
|
/* 关闭电源使能 */
|
||||||
|
rtc_gpio_set_level(PWR_EN_GPIO, 0);
|
||||||
|
rtc_gpio_hold_dis(PWR_EN_GPIO);
|
||||||
|
|
||||||
|
// 确保所有外设已关闭
|
||||||
|
vTaskDelay(200 / portTICK_PERIOD_MS);
|
||||||
|
ESP_LOGI(TAG, "Initiating deep sleep");
|
||||||
|
|
||||||
|
esp_deep_sleep_start();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
ESP_LOGD(TAG, "State changed to %d", static_cast<int>(newState));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
void SetPowerState(PowerState newState) {
|
||||||
|
power_controller_->SetState(newState);
|
||||||
|
}
|
||||||
|
|
||||||
void OnLowBatteryStatusChanged(std::function<void(bool)> callback) {
|
void OnLowBatteryStatusChanged(std::function<void(bool)> callback) {
|
||||||
on_low_battery_status_changed_ = callback;
|
on_low_battery_status_changed_ = callback;
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user