ESP32-PaxCounter/src/power.cpp

403 lines
12 KiB
C++
Raw Normal View History

2019-09-07 19:52:25 +02:00
// Basic config
#include "globals.h"
#include "power.h"
2019-09-07 23:10:53 +02:00
// Local logging tag
static const char TAG[] = __FILE__;
2019-09-07 19:52:25 +02:00
uint8_t batt_level = 0; // display value
#ifdef BAT_MEASURE_ADC
esp_adc_cal_characteristics_t *adc_characs =
(esp_adc_cal_characteristics_t *)calloc(
1, sizeof(esp_adc_cal_characteristics_t));
2019-10-16 21:14:34 +02:00
#ifndef BAT_MEASURE_ADC_UNIT // ADC1
static const adc1_channel_t adc_channel = BAT_MEASURE_ADC;
#else // ADC2
static const adc2_channel_t adc_channel = BAT_MEASURE_ADC;
2021-02-13 15:58:56 +01:00
RTC_NOINIT_ATTR uint64_t RTC_reg_b;
#endif
static const adc_atten_t atten = ADC_ATTEN_DB_11;
static const adc_unit_t unit = ADC_UNIT_1;
#endif // BAT_MEASURE_ADC
2019-09-07 19:52:25 +02:00
#ifdef HAS_PMU
2019-09-09 12:06:23 +02:00
AXP20X_Class pmu;
2019-09-07 19:52:25 +02:00
2019-10-16 21:14:34 +02:00
void AXP192_powerevent_IRQ(void) {
2019-09-30 12:35:03 +02:00
pmu.readIRQ();
if (pmu.isVbusOverVoltageIRQ())
ESP_LOGI(TAG, "USB voltage %.2fV too high.", pmu.getVbusVoltage() / 1000);
if (pmu.isVbusPlugInIRQ())
ESP_LOGI(TAG, "USB plugged, %.2fV @ %.0mA", pmu.getVbusVoltage() / 1000,
pmu.getVbusCurrent());
if (pmu.isVbusRemoveIRQ())
ESP_LOGI(TAG, "USB unplugged.");
if (pmu.isBattPlugInIRQ())
ESP_LOGI(TAG, "Battery is connected.");
if (pmu.isBattRemoveIRQ())
ESP_LOGI(TAG, "Battery was removed.");
if (pmu.isChargingIRQ())
ESP_LOGI(TAG, "Battery charging.");
if (pmu.isChargingDoneIRQ())
ESP_LOGI(TAG, "Battery charging done.");
if (pmu.isBattTempLowIRQ())
ESP_LOGI(TAG, "Battery high temperature.");
if (pmu.isBattTempHighIRQ())
ESP_LOGI(TAG, "Battery low temperature.");
2021-01-30 17:54:34 +01:00
if (pmu.isLowVoltageLevel1IRQ()) {
ESP_LOGI(TAG, "Battery has reached voltage level 1.");
pmu.setChgLEDMode(AXP20X_LED_BLINK_4HZ);
}
if (pmu.isLowVoltageLevel2IRQ()) {
ESP_LOGI(TAG, "Battery has reached voltage level 2.");
pmu.setChgLEDMode(AXP20X_LED_BLINK_1HZ);
}
2019-09-30 12:35:03 +02:00
2020-11-26 21:32:04 +01:00
// short press -> esp32 deep sleep mode, can be exited by pressing user button
2020-12-09 10:15:12 +01:00
if (pmu.isPEKShortPressIRQ()) {
2019-10-16 21:14:34 +02:00
enter_deepsleep(0, HAS_BUTTON);
}
// long press -> shutdown power, can be exited by another longpress
2019-09-30 12:35:03 +02:00
if (pmu.isPEKLongtPressIRQ()) {
AXP192_power(pmu_power_off); // switch off Lora, GPS, display
2019-09-30 12:35:03 +02:00
}
2019-09-30 12:35:03 +02:00
pmu.clearIRQ();
// refresh stored voltage value
2020-04-11 21:30:09 +02:00
read_battlevel();
}
void AXP192_power(pmu_power_t powerlevel) {
switch (powerlevel) {
case pmu_power_off:
2020-12-16 18:32:59 +01:00
pmu.shutdown();
break;
case pmu_power_sleep:
2020-12-16 18:32:59 +01:00
#ifdef PMU_LED_SLEEP_MODE
pmu.setChgLEDMode(PMU_LED_SLEEP_MODE);
#else
2020-12-04 19:09:35 +01:00
pmu.setChgLEDMode(AXP20X_LED_OFF);
2020-12-16 18:32:59 +01:00
#endif
2020-12-24 18:20:53 +01:00
// we don't cut off DCDC1, because OLED display will then block i2c bus
// pmu.setPowerOutPut(AXP192_DCDC1, AXP202_OFF); // OLED off
pmu.setPowerOutPut(AXP192_LDO3, AXP202_OFF); // gps off
pmu.setPowerOutPut(AXP192_LDO2, AXP202_OFF); // lora off
break;
2020-12-16 18:32:59 +01:00
case pmu_power_on:
2020-12-24 18:20:53 +01:00
default:
pmu.setPowerOutPut(AXP192_LDO2, AXP202_ON); // Lora on T-Beam V1.0/1.1
pmu.setPowerOutPut(AXP192_LDO3, AXP202_ON); // Gps on T-Beam V1.0/1.1
pmu.setPowerOutPut(AXP192_DCDC1, AXP202_ON); // OLED on T-Beam v1.0/1.1
pmu.setPowerOutPut(AXP192_DCDC2, AXP202_OFF); // unused on T-Beam v1.0/1.1
pmu.setPowerOutPut(AXP192_EXTEN, AXP202_OFF); // unused on T-Beam v1.0/1.1
2020-12-16 18:32:59 +01:00
#ifdef PMU_LED_RUN_MODE
pmu.setChgLEDMode(PMU_LED_RUN_MODE);
#else
pmu.setChgLEDMode(AXP20X_LED_LOW_LEVEL);
2020-12-16 18:32:59 +01:00
#endif
break;
}
}
void AXP192_showstatus(void) {
2019-09-30 12:35:03 +02:00
if (pmu.isBatteryConnect())
if (pmu.isChargeing())
ESP_LOGI(TAG, "Battery charging, %.2fV @ %.0fmAh",
pmu.getBattVoltage() / 1000, pmu.getBattChargeCurrent());
else
2019-09-30 12:35:03 +02:00
ESP_LOGI(TAG, "Battery not charging");
else
ESP_LOGI(TAG, "No Battery");
if (pmu.isVBUSPlug())
ESP_LOGI(TAG, "USB powered, %.0fmW",
pmu.getVbusVoltage() / 1000 * pmu.getVbusCurrent());
else
ESP_LOGI(TAG, "USB not present");
}
2019-09-07 23:10:53 +02:00
void AXP192_init(void) {
2019-09-08 12:55:27 +02:00
2019-10-12 16:00:22 +02:00
if (pmu.begin(i2c_readBytes, i2c_writeBytes, AXP192_PRIMARY_ADDRESS) ==
AXP_FAIL)
2019-09-30 12:35:03 +02:00
ESP_LOGI(TAG, "AXP192 PMU initialization failed");
else {
2019-09-09 12:06:23 +02:00
2020-12-24 16:52:28 +01:00
// configure voltages
pmu.setDCDC1Voltage(3300); // for external OLED display
pmu.setLDO2Voltage(3300); // LORA VDD 3v3
pmu.setLDO3Voltage(3300); // GPS VDD 3v3
2021-01-30 17:54:34 +01:00
// configure voltage warning levels
pmu.setVWarningLevel1(3600);
pmu.setVWarningLevel2(3800);
pmu.setPowerDownVoltage(3300);
2020-12-24 16:52:28 +01:00
// configure PEK button settings
pmu.setTimeOutShutdown(false); // forced shutdown by PEK enabled
pmu.setShutdownTime(
2021-01-30 17:54:34 +01:00
AXP_POWER_OFF_TIME_6S); // 6 sec button press for shutdown
2020-12-24 16:52:28 +01:00
pmu.setlongPressTime(
AXP_LONGPRESS_TIME_1S5); // 1.5 sec button press for long press
pmu.setStartupTime(
AXP192_STARTUP_TIME_1S); // 1 sec button press for startup
// set battery temperature sensing pin off to save power
pmu.setTSmode(AXP_TS_PIN_MODE_DISABLE);
2019-09-09 12:06:23 +02:00
2019-09-30 12:35:03 +02:00
// switch ADCs on
pmu.adc1Enable(AXP202_BATT_VOL_ADC1, true);
pmu.adc1Enable(AXP202_BATT_CUR_ADC1, true);
pmu.adc1Enable(AXP202_VBUS_VOL_ADC1, true);
pmu.adc1Enable(AXP202_VBUS_CUR_ADC1, true);
2019-09-21 08:23:39 +02:00
#ifdef PMU_INT
2019-09-30 12:35:03 +02:00
pinMode(PMU_INT, INPUT_PULLUP);
attachInterrupt(digitalPinToInterrupt(PMU_INT), PMUIRQ, FALLING);
pmu.enableIRQ(AXP202_VBUS_REMOVED_IRQ | AXP202_VBUS_CONNECT_IRQ |
AXP202_BATT_REMOVED_IRQ | AXP202_BATT_CONNECT_IRQ |
2020-12-24 16:52:28 +01:00
AXP202_CHARGING_FINISHED_IRQ | AXP202_PEK_SHORTPRESS_IRQ,
2019-09-30 12:35:03 +02:00
1);
pmu.clearIRQ();
#endif // PMU_INT
2019-09-09 12:06:23 +02:00
2020-05-09 22:48:29 +02:00
// set charging parameterss according to user settings if we have (see power.h)
#ifdef PMU_CHG_CURRENT
pmu.setChargeControlCur(PMU_CHG_CURRENT);
pmu.setChargingTargetVoltage(PMU_CHG_CUTOFF);
2020-05-09 13:42:08 +02:00
pmu.enableChargeing(true);
#endif
2020-12-16 18:54:23 +01:00
// switch power rails on
AXP192_power(pmu_power_on);
2019-09-30 12:35:03 +02:00
ESP_LOGI(TAG, "AXP192 PMU initialized");
}
}
2019-09-07 23:10:53 +02:00
#endif // HAS_PMU
void calibrate_voltage(void) {
#ifdef BAT_MEASURE_ADC
2019-09-08 12:55:27 +02:00
// configure ADC
#ifndef BAT_MEASURE_ADC_UNIT // ADC1
2020-12-21 19:41:25 +01:00
adc1_config_width(ADC_WIDTH_BIT_12);
adc1_config_channel_atten(adc_channel, atten);
2019-09-08 12:55:27 +02:00
#else // ADC2
2020-12-21 19:41:25 +01:00
adc2_config_channel_atten(adc_channel, atten);
2021-02-13 15:58:56 +01:00
// ADC2 wifi bug workaround, see
// https://github.com/espressif/arduino-esp32/issues/102
RTC_reg_b = READ_PERI_REG(SENS_SAR_READ_CTRL2_REG);
2019-09-08 12:55:27 +02:00
#endif
2019-09-07 23:10:53 +02:00
// calibrate ADC
esp_adc_cal_value_t val_type = esp_adc_cal_characterize(
unit, atten, ADC_WIDTH_BIT_12, DEFAULT_VREF, adc_characs);
// show ADC characterization base
if (val_type == ESP_ADC_CAL_VAL_EFUSE_TP) {
ESP_LOGI(TAG,
"ADC characterization based on Two Point values stored in eFuse");
} else if (val_type == ESP_ADC_CAL_VAL_EFUSE_VREF) {
ESP_LOGI(TAG,
"ADC characterization based on reference voltage stored in eFuse");
} else {
ESP_LOGI(TAG, "ADC characterization based on default reference voltage");
}
#endif
}
2020-04-11 21:30:09 +02:00
uint16_t read_voltage(void) {
2019-09-07 23:10:53 +02:00
uint16_t voltage = 0;
#ifdef HAS_PMU
2020-04-11 21:30:09 +02:00
voltage = pmu.getBattVoltage();
2019-09-07 23:10:53 +02:00
#else
#ifdef BAT_MEASURE_ADC
// multisample ADC
uint32_t adc_reading = 0;
2019-09-08 12:55:27 +02:00
#ifndef BAT_MEASURE_ADC_UNIT // ADC1
2019-09-25 15:00:52 +02:00
for (int i = 0; i < NO_OF_SAMPLES; i++) {
2019-09-07 23:10:53 +02:00
adc_reading += adc1_get_raw(adc_channel);
2019-09-25 15:00:52 +02:00
}
2019-09-08 12:55:27 +02:00
#else // ADC2
2019-09-25 15:00:52 +02:00
int adc_buf = 0;
for (int i = 0; i < NO_OF_SAMPLES; i++) {
2021-02-13 15:58:56 +01:00
// ADC2 wifi bug workaround, see
// https://github.com/espressif/arduino-esp32/issues/102
WRITE_PERI_REG(SENS_SAR_READ_CTRL2_REG, RTC_reg_b);
SET_PERI_REG_MASK(SENS_SAR_READ_CTRL2_REG, SENS_SAR2_DATA_INV);
2020-12-21 19:41:25 +01:00
adc2_get_raw(adc_channel, ADC_WIDTH_BIT_12, &adc_buf);
2019-09-08 12:55:27 +02:00
adc_reading += adc_buf;
2019-09-07 23:10:53 +02:00
}
2019-10-12 16:00:22 +02:00
#endif // BAT_MEASURE_ADC_UNIT
2019-09-07 23:10:53 +02:00
adc_reading /= NO_OF_SAMPLES;
// Convert ADC reading to voltage in mV
voltage = esp_adc_cal_raw_to_voltage(adc_reading, adc_characs);
2019-10-12 16:00:22 +02:00
#endif // BAT_MEASURE_ADC
2019-09-07 23:10:53 +02:00
#ifdef BAT_VOLTAGE_DIVIDER
voltage *= BAT_VOLTAGE_DIVIDER;
2019-09-08 12:55:27 +02:00
#endif // BAT_VOLTAGE_DIVIDER
2019-09-07 23:10:53 +02:00
#endif // HAS_PMU
2020-04-11 21:30:09 +02:00
return voltage;
}
2020-04-10 23:26:29 +02:00
uint8_t read_battlevel(mapFn_t mapFunction) {
2020-04-14 00:15:51 +02:00
// returns the estimated battery level in values 0 ... 100 [percent]
2021-10-11 09:59:41 +02:00
uint8_t batt_percent = 0;
#ifdef HAS_IP5306
2021-10-11 09:59:41 +02:00
batt_percent = IP5306_GetBatteryLevel();
#else
const uint16_t batt_voltage = read_voltage();
if (batt_voltage <= BAT_MIN_VOLTAGE)
2021-10-11 09:59:41 +02:00
batt_percent = 0;
else if (batt_voltage >= BAT_MAX_VOLTAGE)
2021-10-11 09:59:41 +02:00
batt_percent = 100;
else
2021-10-11 09:59:41 +02:00
batt_percent =
(*mapFunction)(batt_voltage, BAT_MIN_VOLTAGE, BAT_MAX_VOLTAGE);
2020-05-02 17:24:52 +02:00
#endif
2021-10-11 09:59:41 +02:00
#if (HAS_LORA)
// set the battery status value to send by LMIC in MAC Command
// DevStatusAns. Available defines in lorabase.h:
// MCMD_DEVS_EXT_POWER = 0x00, // external power supply
// MCMD_DEVS_BATT_MIN = 0x01, // min battery value
// MCMD_DEVS_BATT_MAX = 0xFE, // max battery value
// MCMD_DEVS_BATT_NOINFO = 0xFF, // unknown battery level
// we calculate the applicable value from MCMD_DEVS_BATT_MIN to
// MCMD_DEVS_BATT_MAX from batt_percent value
if (batt_percent == 0)
LMIC_setBatteryLevel(MCMD_DEVS_BATT_NOINFO);
else
LMIC_setBatteryLevel(batt_percent / 100.0 *
(MCMD_DEVS_BATT_MAX - MCMD_DEVS_BATT_MIN + 1));
// overwrite calculated value if we have external power
#ifdef HAS_PMU
if (pmu.isVBUSPlug())
LMIC_setBatteryLevel(MCMD_DEVS_EXT_POWER);
#elif defined HAS_IP5306
if (IP5306_GetPowerSource())
LMIC_setBatteryLevel(MCMD_DEVS_EXT_POWER);
#endif // HAS_PMU
#endif // HAS_LORA
return batt_percent;
2020-04-11 21:30:09 +02:00
}
bool batt_sufficient() {
#if (defined HAS_PMU || defined BAT_MEASURE_ADC || defined HAS_IP5306)
if (batt_level) // we have a battery voltage
return (batt_level > OTA_MIN_BATT);
else
2020-04-11 21:30:09 +02:00
#endif
return true; // we don't know batt level
2020-04-11 21:30:09 +02:00
}
#ifdef HAS_IP5306
// IP5306 code snippet was taken from
// https://gist.github.com/me-no-dev/7702f08dd578de5efa47caf322250b57
#define IP5306_REG_SYS_0 0x00
#define IP5306_REG_SYS_1 0x01
#define IP5306_REG_SYS_2 0x02
#define IP5306_REG_CHG_0 0x20
#define IP5306_REG_CHG_1 0x21
#define IP5306_REG_CHG_2 0x22
#define IP5306_REG_CHG_3 0x23
#define IP5306_REG_CHG_4 0x24
#define IP5306_REG_READ_0 0x70
#define IP5306_REG_READ_1 0x71
#define IP5306_REG_READ_2 0x72
#define IP5306_REG_READ_3 0x77
#define IP5306_REG_READ_4 0x78
#define IP5306_LEDS2PCT(byte) \
((byte & 0x01 ? 25 : 0) + (byte & 0x02 ? 25 : 0) + (byte & 0x04 ? 25 : 0) + \
(byte & 0x08 ? 25 : 0))
uint8_t ip5306_get_bits(uint8_t reg, uint8_t index, uint8_t bits) {
uint8_t value;
2020-05-01 12:24:52 +02:00
if (i2c_readBytes(IP5306_PRIMARY_ADDRESS, reg, &value, 1) == 0xff) {
ESP_LOGW(TAG, "IP5306 get bits fail: 0x%02x", reg);
return 0;
}
return (value >> index) & ((1 << bits) - 1);
}
void ip5306_set_bits(uint8_t reg, uint8_t index, uint8_t bits, uint8_t value) {
uint8_t mask = (1 << bits) - 1, v;
2020-05-01 12:24:52 +02:00
if (i2c_readBytes(IP5306_PRIMARY_ADDRESS, reg, &v, 1) == 0xff) {
ESP_LOGW(TAG, "IP5306 register read fail: 0x%02x", reg);
return;
}
v &= ~(mask << index);
v |= ((value & mask) << index);
2020-05-01 12:24:52 +02:00
if (i2c_writeBytes(IP5306_PRIMARY_ADDRESS, reg, &v, 1) == 0xff)
ESP_LOGW(TAG, "IP5306 register write fail: 0x%02x", reg);
}
uint8_t IP5306_GetPowerSource(void) {
2020-05-01 12:24:52 +02:00
return ip5306_get_bits(IP5306_REG_READ_0, 3, 1); // 0:BAT, 1:VIN
}
uint8_t IP5306_GetBatteryFull(void) {
2020-05-01 12:24:52 +02:00
return ip5306_get_bits(IP5306_REG_READ_1, 3, 1); // 0:CHG/DIS, 1:FULL
}
uint8_t IP5306_GetBatteryLevel(void) {
uint8_t state = (~ip5306_get_bits(IP5306_REG_READ_4, 4, 4)) & 0x0F;
// LED[0-4] State (inverted)
return IP5306_LEDS2PCT(state);
}
2020-05-09 22:48:29 +02:00
void IP5306_SetChargerEnabled(uint8_t v) {
ip5306_set_bits(IP5306_REG_SYS_0, 4, 1, v); // 0:dis,*1:en
}
void IP5306_SetChargeCutoffVoltage(uint8_t v) {
ip5306_set_bits(IP5306_REG_CHG_2, 2, 2,
v); //*0:4.2V, 1:4.3V, 2:4.35V, 3:4.4V
}
void IP5306_SetEndChargeCurrentDetection(uint8_t v) {
ip5306_set_bits(IP5306_REG_CHG_1, 6, 2,
v); // 0:200mA, 1:400mA, *2:500mA, 3:600mA
}
void printIP5306Stats(void) {
bool usb = IP5306_GetPowerSource();
bool full = IP5306_GetBatteryFull();
uint8_t level = IP5306_GetBatteryLevel();
ESP_LOGI(TAG,
"IP5306: Power Source: %s, Battery State: %s, Battery Level: %u%%",
usb ? "USB" : "BATTERY",
full ? "CHARGED" : (usb ? "CHARGING" : "DISCHARGING"), level);
}
2020-05-09 22:48:29 +02:00
void IP5306_init(void) {
IP5306_SetChargerEnabled(1);
IP5306_SetChargeCutoffVoltage(PMU_CHG_CUTOFF);
IP5306_SetEndChargeCurrentDetection(PMU_CHG_CURRENT);
}
#endif // HAS_IP5306