#include "TdeckPower.h" #include #define TAG "power" /** * 2.0 ratio, but +.11 added as display voltage sag compensation. */ #define ADC_MULTIPLIER 2.11 #define ADC_REF_VOLTAGE 3.3f #define BATTERY_VOLTAGE_MIN 3.2f #define BATTERY_VOLTAGE_MAX 4.2f static adc_oneshot_unit_init_cfg_t adcConfig = { .unit_id = ADC_UNIT_1, .clk_src = ADC_RTC_CLK_SRC_DEFAULT, .ulp_mode = ADC_ULP_MODE_DISABLE, }; static adc_oneshot_chan_cfg_t adcChannelConfig = { .atten = ADC_ATTEN_DB_12, .bitwidth = ADC_BITWIDTH_DEFAULT, }; static uint8_t estimateChargeLevelFromVoltage(uint32_t milliVolt) { float volts = std::min((float)milliVolt / 1000.f, BATTERY_VOLTAGE_MAX); float voltage_percentage = (volts - BATTERY_VOLTAGE_MIN) / (BATTERY_VOLTAGE_MAX - BATTERY_VOLTAGE_MIN); float voltage_factor = std::min(1.0f, voltage_percentage); auto charge_level = (uint8_t) (voltage_factor * 100.f); TT_LOG_V(TAG, "mV = %lu, scaled = %.2f, factor = %.2f, result = %d", milliVolt, volts, voltage_factor, charge_level); return charge_level; } TdeckPower::TdeckPower() { if (adc_oneshot_new_unit(&adcConfig, &adcHandle) != ESP_OK) { TT_LOG_E(TAG, "ADC config failed"); return; } if (adc_oneshot_config_channel(adcHandle, ADC_CHANNEL_3, &adcChannelConfig) != ESP_OK) { TT_LOG_E(TAG, "ADC channel config failed"); adc_oneshot_del_unit(adcHandle); return; } } TdeckPower::~TdeckPower() { if (adcHandle) { adc_oneshot_del_unit(adcHandle); } } bool TdeckPower::supportsMetric(MetricType type) const { switch (type) { using enum MetricType; case BatteryVoltage: case ChargeLevel: return true; default: return false; } return false; // Safety guard for when new enum values are introduced } bool TdeckPower::getMetric(MetricType type, MetricData& data) { switch (type) { using enum MetricType; case BatteryVoltage: return readBatteryVoltageSampled(data.valueAsUint32); case ChargeLevel: if (readBatteryVoltageSampled(data.valueAsUint32)) { data.valueAsUint32 = estimateChargeLevelFromVoltage(data.valueAsUint32); return true; } else { return false; } default: return false; } return false; // Safety guard for when new enum values are introduced } bool TdeckPower::readBatteryVoltageOnce(uint32_t& output) { int raw; if (adc_oneshot_read(adcHandle, ADC_CHANNEL_3, &raw) == ESP_OK) { output = ADC_MULTIPLIER * ((1000.f * ADC_REF_VOLTAGE) / 4096.f) * (float)raw; TT_LOG_V(TAG, "Raw = %d, voltage = %lu", raw, output); return true; } else { TT_LOG_E(TAG, "Read failed"); return false; } } #define MAX_VOLTAGE_SAMPLES 15 bool TdeckPower::readBatteryVoltageSampled(uint32_t& output) { size_t samples_read = 0; uint32_t sample_accumulator = 0; uint32_t sample_read_buffer; for (size_t i = 0; i < MAX_VOLTAGE_SAMPLES; ++i) { if (readBatteryVoltageOnce(sample_read_buffer)) { sample_accumulator += sample_read_buffer; samples_read++; } } if (samples_read > 0) { output = sample_accumulator / samples_read; return true; } else { return false; } } static std::shared_ptr power; std::shared_ptr tdeck_get_power() { if (power == nullptr) { power = std::make_shared(); } return power; }