forked from xiaozhi/xiaozhi-esp32
167 lines
5.5 KiB
C++
167 lines
5.5 KiB
C++
#ifndef __POWER_MANAGER_H__
|
||
#define __POWER_MANAGER_H__
|
||
|
||
#include <driver/gpio.h>
|
||
#include <esp_adc/adc_oneshot.h>
|
||
#include <esp_log.h>
|
||
#include <esp_timer.h>
|
||
|
||
class PowerManager {
|
||
private:
|
||
// 电池电量区间-分压电阻为2个100k
|
||
static constexpr struct {
|
||
uint16_t adc;
|
||
uint8_t level;
|
||
} BATTERY_LEVELS[] = {{1980, 0}, {2519, 100}};
|
||
static constexpr size_t BATTERY_LEVELS_COUNT = 2;
|
||
static constexpr size_t ADC_VALUES_COUNT = 10;
|
||
|
||
esp_timer_handle_t timer_handle_ = nullptr;
|
||
gpio_num_t charging_pin_;
|
||
gpio_num_t bat_led_pin_;
|
||
adc_unit_t adc_unit_;
|
||
adc_channel_t adc_channel_;
|
||
uint16_t adc_values_[ADC_VALUES_COUNT];
|
||
size_t adc_values_index_ = 0;
|
||
size_t adc_values_count_ = 0;
|
||
uint8_t battery_level_ = 100;
|
||
bool is_charging_ = false;
|
||
|
||
static constexpr uint8_t MAX_CHANGE_COUNT = 8;
|
||
static constexpr uint32_t TIME_LIMIT = 2000000; // 2 seconds in microseconds
|
||
|
||
uint8_t change_count_ = 0; // 记录状态变化次数
|
||
uint64_t last_change_time_ = 0; // 最后一次状态变化的时间戳(微秒)
|
||
|
||
adc_oneshot_unit_handle_t adc_handle_;
|
||
|
||
void CheckBatteryStatus() {
|
||
uint64_t current_time = esp_timer_get_time(); // 获取当前时间(微秒)
|
||
|
||
// 如果时间间隔超过2秒,则重置状态变化计数
|
||
if (current_time - last_change_time_ > TIME_LIMIT) {
|
||
change_count_ = 0;
|
||
}
|
||
|
||
if (change_count_ < MAX_CHANGE_COUNT) {
|
||
bool new_is_charging = gpio_get_level(bat_led_pin_) != 0; // 检查LED引脚状态
|
||
|
||
// 判断充电引脚状态
|
||
if (new_is_charging) {
|
||
new_is_charging = gpio_get_level(charging_pin_) == 1;
|
||
}
|
||
|
||
// 如果状态有变化
|
||
if (new_is_charging != is_charging_) {
|
||
is_charging_ = new_is_charging;
|
||
change_count_++; // 增加变化次数
|
||
last_change_time_ = current_time; // 更新最后变化时间
|
||
}
|
||
}
|
||
|
||
ReadBatteryAdcData();
|
||
}
|
||
void ReadBatteryAdcData() {
|
||
int adc_value;
|
||
ESP_ERROR_CHECK(adc_oneshot_read(adc_handle_, adc_channel_, &adc_value));
|
||
|
||
adc_values_[adc_values_index_] = adc_value;
|
||
adc_values_index_ = (adc_values_index_ + 1) % ADC_VALUES_COUNT;
|
||
if (adc_values_count_ < ADC_VALUES_COUNT) {
|
||
adc_values_count_++;
|
||
}
|
||
|
||
uint32_t average_adc = 0;
|
||
for (size_t i = 0; i < adc_values_count_; i++) {
|
||
average_adc += adc_values_[i];
|
||
}
|
||
average_adc /= adc_values_count_;
|
||
|
||
CalculateBatteryLevel(average_adc);
|
||
|
||
|
||
// ESP_LOGI("PowerManager", "ADC值: %d 平均值: %ld 电量: %u%%", adc_value, average_adc,
|
||
// battery_level_);
|
||
}
|
||
|
||
void CalculateBatteryLevel(uint32_t average_adc) {
|
||
if (average_adc <= BATTERY_LEVELS[0].adc) {
|
||
battery_level_ = 0;
|
||
} else if (average_adc >= BATTERY_LEVELS[BATTERY_LEVELS_COUNT - 1].adc) {
|
||
battery_level_ = 100;
|
||
} else {
|
||
float ratio = static_cast<float>(average_adc - BATTERY_LEVELS[0].adc) /
|
||
(BATTERY_LEVELS[1].adc - BATTERY_LEVELS[0].adc);
|
||
battery_level_ = ratio * 100;
|
||
}
|
||
}
|
||
|
||
public:
|
||
PowerManager(gpio_num_t charging_pin, gpio_num_t bat_led_pin, adc_unit_t adc_unit = ADC_UNIT_2,
|
||
adc_channel_t adc_channel = ADC_CHANNEL_3)
|
||
: charging_pin_(charging_pin), bat_led_pin_(bat_led_pin), adc_unit_(adc_unit), adc_channel_(adc_channel) {
|
||
|
||
// 配置充电引脚
|
||
gpio_config_t io_conf = {};
|
||
io_conf.intr_type = GPIO_INTR_DISABLE;
|
||
io_conf.mode = GPIO_MODE_INPUT;
|
||
io_conf.pin_bit_mask = (1ULL << charging_pin_);
|
||
io_conf.pull_down_en = GPIO_PULLDOWN_DISABLE;
|
||
io_conf.pull_up_en = GPIO_PULLUP_ENABLE;
|
||
gpio_config(&io_conf);
|
||
|
||
// 配置状态引脚
|
||
io_conf.pull_up_en = GPIO_PULLUP_DISABLE;
|
||
io_conf.pin_bit_mask = (1ULL << bat_led_pin_);
|
||
gpio_config(&io_conf);
|
||
|
||
// 定时器配置
|
||
esp_timer_create_args_t timer_args = {
|
||
.callback =
|
||
[](void* arg) {
|
||
PowerManager* self = static_cast<PowerManager*>(arg);
|
||
self->CheckBatteryStatus();
|
||
},
|
||
.arg = this,
|
||
.dispatch_method = ESP_TIMER_TASK,
|
||
.name = "battery_check_timer",
|
||
.skip_unhandled_events = true,
|
||
};
|
||
ESP_ERROR_CHECK(esp_timer_create(&timer_args, &timer_handle_));
|
||
ESP_ERROR_CHECK(esp_timer_start_periodic(timer_handle_, 500000)); // 1秒
|
||
|
||
// 初始化ADC
|
||
InitializeAdc();
|
||
}
|
||
|
||
void InitializeAdc() {
|
||
adc_oneshot_unit_init_cfg_t init_config = {
|
||
.unit_id = adc_unit_,
|
||
.ulp_mode = ADC_ULP_MODE_DISABLE,
|
||
};
|
||
ESP_ERROR_CHECK(adc_oneshot_new_unit(&init_config, &adc_handle_));
|
||
|
||
adc_oneshot_chan_cfg_t chan_config = {
|
||
.atten = ADC_ATTEN_DB_12,
|
||
.bitwidth = ADC_BITWIDTH_12,
|
||
};
|
||
|
||
ESP_ERROR_CHECK(adc_oneshot_config_channel(adc_handle_, adc_channel_, &chan_config));
|
||
}
|
||
|
||
~PowerManager() {
|
||
if (timer_handle_) {
|
||
esp_timer_stop(timer_handle_);
|
||
esp_timer_delete(timer_handle_);
|
||
}
|
||
if (adc_handle_) {
|
||
adc_oneshot_del_unit(adc_handle_);
|
||
}
|
||
}
|
||
|
||
bool IsCharging() { return is_charging_; }
|
||
|
||
uint8_t GetBatteryLevel() { return battery_level_; }
|
||
};
|
||
#endif // __POWER_MANAGER_H__
|