mirror of
https://github.com/meshtastic/firmware.git
synced 2025-08-05 13:14:45 +00:00

Some checks are pending
CI / setup (check) (push) Waiting to run
CI / setup (esp32) (push) Waiting to run
CI / setup (esp32c3) (push) Waiting to run
CI / setup (esp32c6) (push) Waiting to run
CI / setup (esp32s3) (push) Waiting to run
CI / setup (nrf52840) (push) Waiting to run
CI / setup (rp2040) (push) Waiting to run
CI / setup (stm32) (push) Waiting to run
CI / check (push) Blocked by required conditions
CI / build-esp32 (push) Blocked by required conditions
CI / build-esp32-s3 (push) Blocked by required conditions
CI / build-esp32-c3 (push) Blocked by required conditions
CI / build-esp32-c6 (push) Blocked by required conditions
CI / build-nrf52 (push) Blocked by required conditions
CI / build-rpi2040 (push) Blocked by required conditions
CI / build-stm32 (push) Blocked by required conditions
CI / package-raspbian (push) Waiting to run
CI / package-raspbian-armv7l (push) Waiting to run
CI / package-native (push) Waiting to run
CI / build-docker (push) Waiting to run
CI / after-checks (push) Blocked by required conditions
CI / gather-artifacts (esp32) (push) Blocked by required conditions
CI / gather-artifacts (esp32c3) (push) Blocked by required conditions
CI / gather-artifacts (esp32c6) (push) Blocked by required conditions
CI / gather-artifacts (esp32s3) (push) Blocked by required conditions
CI / gather-artifacts (nrf52840) (push) Blocked by required conditions
CI / gather-artifacts (rp2040) (push) Blocked by required conditions
CI / gather-artifacts (stm32) (push) Blocked by required conditions
CI / release-artifacts (push) Blocked by required conditions
CI / release-firmware (esp32) (push) Blocked by required conditions
CI / release-firmware (esp32c3) (push) Blocked by required conditions
CI / release-firmware (esp32c6) (push) Blocked by required conditions
CI / release-firmware (esp32s3) (push) Blocked by required conditions
CI / release-firmware (nrf52840) (push) Blocked by required conditions
CI / release-firmware (rp2040) (push) Blocked by required conditions
CI / release-firmware (stm32) (push) Blocked by required conditions
Flawfinder Scan / Flawfinder (push) Waiting to run
* INA219 charging detection minimal implementation: if there is a configured INA219 sensor for battery monitoring we can take the current flow across the shunt resistor to know if we are charging the battery - negative milliamps indicate charging * Update Power.cpp added comments and 2 extra defines to disable and swap detection direction * Update Power.cpp fix disabled case * move getCurrentMa() to new CurrentSensor class * INA219 charging detection minimal implementation: if there is a configured INA219 sensor for battery monitoring we can take the current flow across the shunt resistor to know if we are charging the battery - negative milliamps indicate charging * Update Power.cpp added comments and 2 extra defines to disable and swap detection direction * Update Power.cpp fix disabled case * move getCurrentMa() to new CurrentSensor class * add INA3221 charging detection * RP2040: Update core; add mDNS support (#5355) * Update arduino-pico core * RP2040: Add mDNS support * SimpleMDNS `begin` now returns a bool * Add `-g` option to `debug_build_flags` to link files for gdb * RAK11310 needs old platform as well * Change defines to specific architecture * Core version 4.2.1 is out * Add sudo to apt-get commands for Raspbian Build (#5364) Without sudo, inadequate permissions to runs the commands meant the build was failing. * Typo fix in build_raspbian.yml (#5365) s/sudp/sudo :(:(:( * Rework some things * Trunk * Separate littlefs bundle * version tags * Diag * Add littlefswebui * Bug fixed in ExternalNotificationModule (#5375) While `nagging` setExternalState wasn't written to Buzzer & Vibra so output was never toggled. Possible fix for #5348 * Cleanup static files from bad Web UI bundle on 2.5.13 release (#5376) * Cleanup static files from bad Web UI bundle on 2.5.13 release * Check existence first * Esp32 is the only one we care about * Move some actions to after `startTransmit()` (#5383) To minimize the time between channel scan and actual transmit * [create-pull-request] automated change (#5380) Co-authored-by: thebentern <9000580+thebentern@users.noreply.github.com> * Allows all 3 PKI keys to be added to userPrefs.h (#4969) and a tool. (#5368) * more userPrefs.h Added PKI Admin keys to userPrefs.h * Update userPrefs.h Allows all 3 PKI keys to be added to userPrefs.h (#4969) * Update NodeDB.cpp Trunk * Update userPrefs.h Changed wording * Create base64_to_hex.py A little tool for converting base64 PKI Keys to decoded byte that userPrefs.h can understand. * more userPrefs.h Added PKI Admin keys to userPrefs.h * Update userPrefs.h Allows all 3 PKI keys to be added to userPrefs.h (#4969) * Update NodeDB.cpp Trunk * Update userPrefs.h Changed wording * Create base64_to_hex.py A little tool for converting base64 PKI Keys to decoded byte that userPrefs.h can understand. * [create-pull-request] automated change (#5388) Co-authored-by: thebentern <9000580+thebentern@users.noreply.github.com> * add smiley emoji (#5391) * add smiley emoji * clang-formatted * Anable trace route function on wismeshtap platform (#5389) * fix 'symbal' typo (#5395) * [create-pull-request] automated change (#5399) Co-authored-by: thebentern <9000580+thebentern@users.noreply.github.com> * /api/v1/fromradio: add OPTIONS handler for CORS. (#5386) This avoids hitting the 404 Not Found handler, which breaks connection keep-alive, so this change fixes a big performance regression for Web Client in Chrome: https://github.com/meshtastic/firmware/issues/5385 Tested on Heltec V3. Co-authored-by: Ben Meadors <benmmeadors@gmail.com> * Make heart emoji usable (#5403) * Create a specific hw_model for WisMesh Tap (#5400) * Create a specific hw_model for WisMesh Tap * Trunk * HAS_ETHERNET * Remove it altogether * Don't need these either * Fix RTC time injection and consolidate position logic (#5396) * Fix RTC time injection and consolidate position logic * Comment out unused var warning * Backerds * Update arduino-pico core to fix sporadic hangs (#5406) * Update platform-raspberrypi also (#5407) * Update arduino-pico core to fix sporadic hangs * Update platform-raspberrypi also * --web added to device-install(.sh/.bat) (#5405) * Add --web * Update device-install.bat Forgot a "-" a few places. --------- Co-authored-by: Ben Meadors <benmmeadors@gmail.com> * add GPS in indicator board (#5411) * Fixed NMEA sentence issue in CalTopo as well as bug with no printing all of the nodes (#5412) * --web littlefswebui-* typo fix (#5416) * Add --web * Update device-install.bat Forgot a "-" a few places. * Typo fix. * Typo fix --------- Co-authored-by: Ben Meadors <benmmeadors@gmail.com> Co-authored-by: GUVWAF <78759985+GUVWAF@users.noreply.github.com> * Temporarily disable MDNS when MQTT is enabled (#5418) Leads to a panic * Check for OkToMqtt flag presence before uplinking to MQTT (#5413) * Check for oktomqtt flag presence before uplinking to MQTT * Move to mqtt->onSend * Temetry can respond to want-response for LocalStats variant (#5414) * Seems like the last DIY board that's not "extra" (#5420) * Cherry pick tdeck fixes (#5422) * Try-fix (workaround) T-Deck audio crash * set T-Deck audio to unused 48 (mem mclk) * swap mclk to gpio 21 * dreamcatcher: assign GPIO44 to audio mclk --------- Co-authored-by: mverch67 <manuel.verch@gmx.de> * add canned message and keyboard in indicator board (#5410) * add canned message and keyboard in indicator board * Added virtual keyboard macro and enabled for Indicator * Cleanup macros by applying USE_VIRTUAL_KEYBOARD and DISPLAY_CLOCK_FRAME --------- Co-authored-by: Ben Meadors <benmmeadors@gmail.com> * Update build-native.sh (#5415) * Update build-native.sh Device-install.sh and device-update.sh are not used on native platform, skip copying to release directory after build and copy native-install.sh and native-run.sh instead. * Update build-native.sh Skip native-run.sh copy * Cleans up visibility in GPS.h (#5426) Signed-off-by: Christopher Hoover <ch@murgatroid.com> * Fix admin key loading from userPrefs.h (#5417) * Fix LED pinout for T-Echo board marked v1.0, date 2021-6-28 * Merge PR #420 * Fixed double and missing Default class. * Use correct format specifier and fixed typo. * Removed duplicate code. * Fix error: #if with no expression * Fix warning: extra tokens at end of #endif directive. * Fix antenna switching logic. Complementary-pin control logic is required on the rp2040-lora board. * Fix deprecated macros. * Set RP2040 in dormant mode when deep sleep is triggered. * Fix array out of bounds read. * Admin key count needs to be set otherwise the key will be zero loaded after reset. * Don't reset the admin key size when loading defaults. Preserve an existing key in config if possible. * Remove log spam when reading INA voltage sensor. * Remove static declaration for admin keys from userPrefs.h. Load hard coded admin keys in case config file has empty slots. * Removed newlines from log. --------- Co-authored-by: Ben Meadors <benmmeadors@gmail.com> Co-authored-by: Thomas Göttgens <tgoettgens@gmail.com> * try to detect dfrobot station to tell it apart from an ublox gps. (#5393) * Remove BMA-423 and STK8X by default (#5429) * Remove BMA-423 by default * STK * Wrong macro * Helps if you include the file * [create-pull-request] automated change (#5431) Co-authored-by: caveman99 <25002+caveman99@users.noreply.github.com> * Support for the ClimateGuard RadSens Geiger-Muller tube (#5425) * fixes https://github.com/meshtastic/firmware/issues/5434 (#5435) * update libpax * fix interval init * Fix memory leaks by adding missing `free()` calls before early returns in `MQTT::onReceive` (#5439) This fix addresses memory leaks in the `MQTT::onReceive` function by ensuring that dynamically allocated resources (`e.channel_id`, `e.gateway_id` and `e.packet`) are properly freed before each early return. Previously, these resources were only freed at the end of the function, leaving them unhandled in certain exit paths. Adding the missing `free()` calls prevents memory leaks and ensures proper resource cleanup in all scenarios. * Removing 1.0 legacy boards from releases and completely removing Heltec wireless capsule from support (#5436) Co-authored-by: Tom Fifield <tom@tomfifield.net> * A second round of cleanup on GPS.h. (#5433) * Move yet more stuff out of GPS.h and into file scope. * Protect code macros from eating semicolons. * Remove unused (and unimplemented) getDOPString. * clang-format with project style file on affected files. Signed-off-by: Christopher Hoover <ch@murgatroid.com> * enable MQTT with TLS on RPi picow (#5442) Co-authored-by: Ben Meadors <benmmeadors@gmail.com> * Don't powersave on Wifi (#5443) * Don't go into light sleep with wifi enabled * Move * Trunk * Revert "Seems like the last DIY board that's not "extra" (#5420)" (#5446) This reverts commite6fb6b115a
. * Actually gunzip all the files when building a .deb (#5449) * [create-pull-request] automated change (#5457) Co-authored-by: thebentern <9000580+thebentern@users.noreply.github.com> * Cleanup i2c scan logs and macro to save some bytes and remain consistent (#5455) * Cleanup i2c scan logs and macro to save some bytes and remain consistent * Functions are better than macros * Exclude i2c scan for STM32 * Useless log * Clean up some inline functions (#5454) * Use isWithinTimespanMs to avoid refererence to NodeDb instance inside of NodeDb (#5453) * fix cors for meshtasticd to allow use of cross origin clients (#5463) * Remove ATECC crypto chip placeholder code (#5461) * GPS.h cleanups round 3. (#5447) * GPS.h cleanups round 3. No effective behavior change. Protected members can be private so make it so. (Supporting subclasses needs a lot more work.) Moves uBloxGnssModelInfo into file scope. Moves uBloxProtocolVersion into uBloxGnssModelInfo. Moves baud rate arrays into file scope. Removes unused/ unimplemented powerStateToString. Signed-off-by: Christopher Hoover <ch@murgatroid.com> * Trunk Format. --------- Signed-off-by: Christopher Hoover <ch@murgatroid.com> Co-authored-by: Tom Fifield <tom@tomfifield.net> * Fix ukrainian fonts (#5468) * FIX: rollback to !4624 * UPDATE: new 16 and 24 UA Fonts and fixes * fix: Solve the lightsleep crash problem via disable lightsleep for indicator. (#5470) * Trunk * Warnings and log cleanup (#5472) * Don't log if keyboard not found * Signed comparison issue * [create-pull-request] automated change (#5475) Co-authored-by: thebentern <9000580+thebentern@users.noreply.github.com> * Adds libusb dev package to Raspbian build steps (#5480) * [create-pull-request] automated change (#5478) Co-authored-by: thebentern <9000580+thebentern@users.noreply.github.com> * Portduino fixes (#5479) * Set config.yaml defaults even if General is missing * Unsigned values should get %u in logging * Update arduino-pico core and remove MDNS restriction (#5483) * Update xiao_esp32 fully support L67K (#5488) L67K module hardware changed * Convert userprefs to a json file instead of header file which has to be included everywhere (#5471) * WIP * Got string quoting and macro expansion working * Need the placeholder * Cleanup * Missed a user prefs reference * Update jsonc * SimRadio: clean-up and emulate collisions (#5487) * Clean up SimRadio and don't let it use PKC * Add collision emulation for SimRadio * Add stats from SimRadio to LocalStats * Make emulating collisions optional * add nodeId to nodeinfo update log lines and removed redundant nodeinfo update log line (#5493) * Refact the macro definition of GPS initialization of GPSDEFAULTD_NOT_PRESENT and added seeeed Indicator to this sequence (#5494) Co-authored-by: Ben Meadors <benmmeadors@gmail.com> * Extend Length of Source and Destination Node IDs Logged (#5492) * show 8 chars for logging source and destination ids * extend legnth of source and destination nodes in log * Added femtofox configs (#5477) * added femtofox configs * Rename bin/config.d/femtofox_Waveshare-SX126X-XXXM_AI-Thinker-RA-01SH.yaml to bin/config.d/femtofox/femtofox_Waveshare-SX126X-XXXM_AI-Thinker-RA-01SH.yaml * moved femtofox configs to subdir * [Add] LR1110, LR1120 and LR1121 to linux native Portduino (#5496) * Update main.cpp * Update PortduinoGlue.h * Update PortduinoGlue.cpp * Update PortduinoGlue.cpp * Update PortduinoGlue.cpp * Update main.cpp * [create-pull-request] automated change (#5500) Co-authored-by: thebentern <9000580+thebentern@users.noreply.github.com> * Fix minor typos in package workflows (#5505) * Don't use channel index for encrypted packet (#5509) * Don't use channel index for encrypted packet * Remove assert in `getKey`, set invalid key length So encrypting will fail without reboot * Reset channel to 0 when unable to encrypt Such that the NAK doesn't use the failing channel hash * Always Announce MDNS meshtastic service (#5503) * refactor server api port into define * always announce MDNS meshtastic service * fix nodeDB erase loop when free mem returns invalid value (0, -1). (#5519) Co-authored-by: mverch67 <manuel.verch@gmx.de> * Add heltec capsule back * Revert "Add heltec capsule back" This reverts commitfc16d93421
. * Lets try this again minus device ui * Add popular nrf52 pro micro to the builds (#5523) * Add MACAddress to config.yaml (#5506) * Add MACAddress to config.yaml * Better error handling on native, including failing to launch with blank MAC Address and real hardware. * Re-arrange Mac Address handling and add MACAddressSource * Bump portduino to remove macaddr function there --------- Co-authored-by: Ben Meadors <benmmeadors@gmail.com> * Configure Seeed Xiao S3 RX enable pin (#5517) * Create OpenWRT_One_mikroBUS_sx1262.yaml (#5529) * tlora_v2_1_16: Unset BUTTON_PIN and BUTTON_NEED_PULLUP (#5535) Unset BUTTON_PIN and BUTTON_NEED_PULLUP as the board ships without a user button. Devices and users expecting a button on GPIO12 have to set [GPIO for user button](https://meshtastic.org/docs/configuration/radio/device/#gpio-for-user-button) to 12 (or any GPIO pin the momentary switch was connected to) to restore functionality. Signed-off-by: Andrew Yong <me@ndoo.sg> * [create-pull-request] automated change (#5530) Co-authored-by: thebentern <9000580+thebentern@users.noreply.github.com> * Fix detection for some RadSens hardware versions (#5542) Co-authored-by: Jake-B <jake-b@users.noreply.github.com> * Initialize dmac array to nulls (#5538) * Initialize dmac array to nulls * Use std::cout for print before console is init. * Update OpenWRT_One_mikroBUS_sx1262.yaml (#5544) * Add portduino-buildroot variant (#5540) * Add portduino-buildroot variant * Update platform-native for platform-buildroot * portduino-buildroot: Define c standard (#5547) * Portduino: Move meshtasticd/web out of /usr/share/doc/ (#5548) * Portduino: fix transitional symlinks (#5550) * Windows Support - Trunk and Platformio (#5397) (#5518) * Add support for GPG * Add usb device support * Add trunk.io to devcontainer * Trunk things * trunk fmt * formatting * fix trivy/DS002, checkov/CKV_DOCKER_3 * hide docker extension popup * fix trivy/DS026, checkov/CKV_DOCKER_2 Co-authored-by: Kalle Lilja <15094562+ThatKalle@users.noreply.github.com> * Synch minor changes from TFT branch (#5520) * Synch minor changes from TFT branch Includes: * New nordicnrf52 minor version (10.5.0 --> 10.6.0) * Optimisations for T_DECK * preparation for MESH_TAB * add ext notification module to portduino --------- Co-authored-by: mverch67 <manuel.verch@gmx.de> * DIO3_TCXO_VOLTAGE in config.yaml can now take an exact voltage (#5558) * Support TLORA_V3.0 (#5563) - Support TLORA_V3.0. Update of the legendary 2.1_1.6.1 with solar charger, TCXO and IPEX connector. - 'extra' some short-lived EOL intermediate boards in that range. If possible use T3S3 instead of all of these! - update trunk to latest version * Create OpenWRT-One-mikroBUS-LR-IOT-CLICK.yaml (#5564) * Portduino: fix setting hwId via argument (#5565) * INA219 charging detection minimal implementation: if there is a configured INA219 sensor for battery monitoring we can take the current flow across the shunt resistor to know if we are charging the battery - negative milliamps indicate charging * Update Power.cpp added comments and 2 extra defines to disable and swap detection direction * Trunk Fixes * Add INA226 support --------- Signed-off-by: Christopher Hoover <ch@murgatroid.com> Signed-off-by: Andrew Yong <me@ndoo.sg> Co-authored-by: Ben Meadors <benmmeadors@gmail.com> Co-authored-by: Jonathan Bennett <jbennett@incomsystems.biz> Co-authored-by: GUVWAF <78759985+GUVWAF@users.noreply.github.com> Co-authored-by: Tom Fifield <tom@tomfifield.net> Co-authored-by: Michael Gjelsø <36234524+gjelsoe@users.noreply.github.com> Co-authored-by: github-actions[bot] <41898282+github-actions[bot]@users.noreply.github.com> Co-authored-by: thebentern <9000580+thebentern@users.noreply.github.com> Co-authored-by: jcyrio <50239349+jcyrio@users.noreply.github.com> Co-authored-by: Daniel.Cao <144674500+DanielCao0@users.noreply.github.com> Co-authored-by: Catalin Patulea <cronos586@gmail.com> Co-authored-by: dylanli <dylan3000dylan@gmail.com> Co-authored-by: mverch67 <manuel.verch@gmx.de> Co-authored-by: madeofstown <33820964+madeofstown@users.noreply.github.com> Co-authored-by: Christopher Hoover <ch@murgatroid.com> Co-authored-by: Mictronics <github@mictronics.de> Co-authored-by: Thomas Göttgens <tgoettgens@gmail.com> Co-authored-by: caveman99 <25002+caveman99@users.noreply.github.com> Co-authored-by: jake-b <1012393+jake-b@users.noreply.github.com> Co-authored-by: César de Tassis Filho <CTassisF@users.noreply.github.com> Co-authored-by: Tomas Dubec <tomas.dubec@gmail.com> Co-authored-by: Liam Cottle <liam@liamcottle.com> Co-authored-by: panaceya <panaceya@users.noreply.github.com> Co-authored-by: virgil <virgil.wang.cj@gmail.com> Co-authored-by: Robert <r@rbrt.io> Co-authored-by: noon92 <40807970+noon92@users.noreply.github.com> Co-authored-by: Mark Trevor Birss <markbirss@gmail.com> Co-authored-by: broglep <20624281+broglep@users.noreply.github.com> Co-authored-by: Matthias Granberry <matthias.granberry@gmail.com> Co-authored-by: Andrew Yong <me@ndoo.sg> Co-authored-by: Jake-B <jake-b@users.noreply.github.com> Co-authored-by: Austin <vidplace7@gmail.com> Co-authored-by: Kalle Lilja <15094562+ThatKalle@users.noreply.github.com>
1193 lines
41 KiB
C++
1193 lines
41 KiB
C++
/**
|
|
* @file Power.cpp
|
|
* @brief This file contains the implementation of the Power class, which is responsible for managing power-related functionality
|
|
* of the device. It includes battery level sensing, power management unit (PMU) control, and power state machine management. The
|
|
* Power class is used by the main device class to manage power-related functionality.
|
|
*
|
|
* The file also includes implementations of various battery level sensors, such as the AnalogBatteryLevel class, which assumes
|
|
* the battery voltage is attached via a voltage-divider to an analog input.
|
|
*
|
|
* This file is part of the Meshtastic project.
|
|
* For more information, see: https://meshtastic.org/
|
|
*/
|
|
#include "power.h"
|
|
#include "NodeDB.h"
|
|
#include "PowerFSM.h"
|
|
#include "Throttle.h"
|
|
#include "buzz/buzz.h"
|
|
#include "configuration.h"
|
|
#include "main.h"
|
|
#include "meshUtils.h"
|
|
#include "sleep.h"
|
|
|
|
// Working USB detection for powered/charging states on the RAK platform
|
|
#ifdef NRF_APM
|
|
#include "nrfx_power.h"
|
|
#endif
|
|
|
|
#if defined(DEBUG_HEAP_MQTT) && !MESHTASTIC_EXCLUDE_MQTT
|
|
#include "mqtt/MQTT.h"
|
|
#include "target_specific.h"
|
|
#if HAS_WIFI
|
|
#include <WiFi.h>
|
|
#endif
|
|
|
|
#endif
|
|
|
|
#ifndef DELAY_FOREVER
|
|
#define DELAY_FOREVER portMAX_DELAY
|
|
#endif
|
|
|
|
#if defined(BATTERY_PIN) && defined(ARCH_ESP32)
|
|
|
|
#ifndef BAT_MEASURE_ADC_UNIT // ADC1 is default
|
|
static const adc1_channel_t adc_channel = ADC_CHANNEL;
|
|
static const adc_unit_t unit = ADC_UNIT_1;
|
|
#else // ADC2
|
|
static const adc2_channel_t adc_channel = ADC_CHANNEL;
|
|
static const adc_unit_t unit = ADC_UNIT_2;
|
|
RTC_NOINIT_ATTR uint64_t RTC_reg_b;
|
|
|
|
#endif // BAT_MEASURE_ADC_UNIT
|
|
|
|
esp_adc_cal_characteristics_t *adc_characs = (esp_adc_cal_characteristics_t *)calloc(1, sizeof(esp_adc_cal_characteristics_t));
|
|
#ifndef ADC_ATTENUATION
|
|
static const adc_atten_t atten = ADC_ATTEN_DB_12;
|
|
#else
|
|
static const adc_atten_t atten = ADC_ATTENUATION;
|
|
#endif
|
|
#endif // BATTERY_PIN && ARCH_ESP32
|
|
|
|
#ifdef EXT_CHRG_DETECT
|
|
#ifndef EXT_CHRG_DETECT_MODE
|
|
static const uint8_t ext_chrg_detect_mode = INPUT;
|
|
#else
|
|
static const uint8_t ext_chrg_detect_mode = EXT_CHRG_DETECT_MODE;
|
|
#endif
|
|
#ifndef EXT_CHRG_DETECT_VALUE
|
|
static const uint8_t ext_chrg_detect_value = HIGH;
|
|
#else
|
|
static const uint8_t ext_chrg_detect_value = EXT_CHRG_DETECT_VALUE;
|
|
#endif
|
|
#endif
|
|
|
|
#if HAS_TELEMETRY && !MESHTASTIC_EXCLUDE_ENVIRONMENTAL_SENSOR && !defined(ARCH_PORTDUINO)
|
|
INA219Sensor ina219Sensor;
|
|
INA226Sensor ina226Sensor;
|
|
INA260Sensor ina260Sensor;
|
|
INA3221Sensor ina3221Sensor;
|
|
#endif
|
|
|
|
#if !MESHTASTIC_EXCLUDE_I2C && !defined(ARCH_PORTDUINO) && !defined(ARCH_STM32WL)
|
|
#include "modules/Telemetry/Sensor/MAX17048Sensor.h"
|
|
#include <utility>
|
|
extern std::pair<uint8_t, TwoWire *> nodeTelemetrySensorsMap[_meshtastic_TelemetrySensorType_MAX + 1];
|
|
#if HAS_TELEMETRY && (!MESHTASTIC_EXCLUDE_ENVIRONMENTAL_SENSOR || !MESHTASTIC_EXCLUDE_POWER_TELEMETRY)
|
|
MAX17048Sensor max17048Sensor;
|
|
#endif
|
|
#endif
|
|
|
|
#if HAS_RAKPROT && !defined(ARCH_PORTDUINO)
|
|
RAK9154Sensor rak9154Sensor;
|
|
#endif
|
|
|
|
#ifdef HAS_PMU
|
|
XPowersLibInterface *PMU = NULL;
|
|
#else
|
|
|
|
// Copy of the base class defined in axp20x.h.
|
|
// I'd rather not include axp20x.h as it brings Wire dependency.
|
|
class HasBatteryLevel
|
|
{
|
|
public:
|
|
/**
|
|
* Battery state of charge, from 0 to 100 or -1 for unknown
|
|
*/
|
|
virtual int getBatteryPercent() { return -1; }
|
|
|
|
/**
|
|
* The raw voltage of the battery or NAN if unknown
|
|
*/
|
|
virtual uint16_t getBattVoltage() { return 0; }
|
|
|
|
/**
|
|
* return true if there is a battery installed in this unit
|
|
*/
|
|
virtual bool isBatteryConnect() { return false; }
|
|
|
|
virtual bool isVbusIn() { return false; }
|
|
virtual bool isCharging() { return false; }
|
|
};
|
|
#endif
|
|
|
|
bool pmu_irq = false;
|
|
|
|
Power *power;
|
|
|
|
using namespace meshtastic;
|
|
|
|
#ifndef AREF_VOLTAGE
|
|
#if defined(ARCH_NRF52)
|
|
/*
|
|
* Internal Reference is +/-0.6V, with an adjustable gain of 1/6, 1/5, 1/4,
|
|
* 1/3, 1/2 or 1, meaning 3.6, 3.0, 2.4, 1.8, 1.2 or 0.6V for the ADC levels.
|
|
*
|
|
* External Reference is VDD/4, with an adjustable gain of 1, 2 or 4, meaning
|
|
* VDD/4, VDD/2 or VDD for the ADC levels.
|
|
*
|
|
* Default settings are internal reference with 1/6 gain (GND..3.6V ADC range)
|
|
*/
|
|
#define AREF_VOLTAGE 3.6
|
|
#else
|
|
#define AREF_VOLTAGE 3.3
|
|
#endif
|
|
#endif
|
|
|
|
/**
|
|
* If this board has a battery level sensor, set this to a valid implementation
|
|
*/
|
|
static HasBatteryLevel *batteryLevel; // Default to NULL for no battery level sensor
|
|
|
|
#ifdef BATTERY_PIN
|
|
|
|
static void adcEnable()
|
|
{
|
|
#ifdef ADC_CTRL // enable adc voltage divider when we need to read
|
|
#ifdef ADC_USE_PULLUP
|
|
pinMode(ADC_CTRL, INPUT_PULLUP);
|
|
#else
|
|
#ifdef HELTEC_V3
|
|
pinMode(ADC_CTRL, INPUT);
|
|
uint8_t adc_ctl_enable_value = !(digitalRead(ADC_CTRL));
|
|
pinMode(ADC_CTRL, OUTPUT);
|
|
digitalWrite(ADC_CTRL, adc_ctl_enable_value);
|
|
#else
|
|
pinMode(ADC_CTRL, OUTPUT);
|
|
digitalWrite(ADC_CTRL, ADC_CTRL_ENABLED);
|
|
#endif
|
|
#endif
|
|
delay(10);
|
|
#endif
|
|
}
|
|
|
|
static void adcDisable()
|
|
{
|
|
#ifdef ADC_CTRL // disable adc voltage divider when we need to read
|
|
#ifdef ADC_USE_PULLUP
|
|
pinMode(ADC_CTRL, INPUT_PULLDOWN);
|
|
#else
|
|
#ifdef HELTEC_V3
|
|
pinMode(ADC_CTRL, ANALOG);
|
|
#else
|
|
digitalWrite(ADC_CTRL, !ADC_CTRL_ENABLED);
|
|
#endif
|
|
#endif
|
|
#endif
|
|
}
|
|
|
|
#endif
|
|
|
|
/**
|
|
* A simple battery level sensor that assumes the battery voltage is attached via a voltage-divider to an analog input
|
|
*/
|
|
class AnalogBatteryLevel : public HasBatteryLevel
|
|
{
|
|
public:
|
|
/**
|
|
* Battery state of charge, from 0 to 100 or -1 for unknown
|
|
*/
|
|
virtual int getBatteryPercent() override
|
|
{
|
|
#if defined(HAS_RAKPROT) && !defined(ARCH_PORTDUINO) && !defined(HAS_PMU)
|
|
if (hasRAK()) {
|
|
return rak9154Sensor.getBusBatteryPercent();
|
|
}
|
|
#endif
|
|
|
|
float v = getBattVoltage();
|
|
|
|
if (v < noBatVolt)
|
|
return -1; // If voltage is super low assume no battery installed
|
|
|
|
#ifdef NO_BATTERY_LEVEL_ON_CHARGE
|
|
// This does not work on a RAK4631 with battery connected
|
|
if (v > chargingVolt)
|
|
return 0; // While charging we can't report % full on the battery
|
|
#endif
|
|
/**
|
|
* @brief Battery voltage lookup table interpolation to obtain a more
|
|
* precise percentage rather than the old proportional one.
|
|
* @author Gabriele Russo
|
|
* @date 06/02/2024
|
|
*/
|
|
float battery_SOC = 0.0;
|
|
uint16_t voltage = v / NUM_CELLS; // single cell voltage (average)
|
|
for (int i = 0; i < NUM_OCV_POINTS; i++) {
|
|
if (OCV[i] <= voltage) {
|
|
if (i == 0) {
|
|
battery_SOC = 100.0; // 100% full
|
|
} else {
|
|
// interpolate between OCV[i] and OCV[i-1]
|
|
battery_SOC = (float)100.0 / (NUM_OCV_POINTS - 1.0) *
|
|
(NUM_OCV_POINTS - 1.0 - i + ((float)voltage - OCV[i]) / (OCV[i - 1] - OCV[i]));
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
return clamp((int)(battery_SOC), 0, 100);
|
|
}
|
|
|
|
/**
|
|
* The raw voltage of the batteryin millivolts or NAN if unknown
|
|
*/
|
|
virtual uint16_t getBattVoltage() override
|
|
{
|
|
|
|
#if defined(HAS_RAKPROT) && !defined(ARCH_PORTDUINO) && !defined(HAS_PMU)
|
|
if (hasRAK()) {
|
|
return getRAKVoltage();
|
|
}
|
|
#endif
|
|
|
|
#if HAS_TELEMETRY && !defined(ARCH_PORTDUINO) && !defined(ARCH_STM32WL) && !defined(HAS_PMU) && \
|
|
!MESHTASTIC_EXCLUDE_ENVIRONMENTAL_SENSOR
|
|
if (hasINA()) {
|
|
return getINAVoltage();
|
|
}
|
|
#endif
|
|
|
|
#ifndef ADC_MULTIPLIER
|
|
#define ADC_MULTIPLIER 2.0
|
|
#endif
|
|
|
|
#ifndef BATTERY_SENSE_SAMPLES
|
|
#define BATTERY_SENSE_SAMPLES \
|
|
15 // Set the number of samples, it has an effect of increasing sensitivity in complex electromagnetic environment.
|
|
#endif
|
|
|
|
#ifdef BATTERY_PIN
|
|
// Override variant or default ADC_MULTIPLIER if we have the override pref
|
|
float operativeAdcMultiplier =
|
|
config.power.adc_multiplier_override > 0 ? config.power.adc_multiplier_override : ADC_MULTIPLIER;
|
|
// Do not call analogRead() often.
|
|
const uint32_t min_read_interval = 5000;
|
|
if (!initial_read_done || !Throttle::isWithinTimespanMs(last_read_time_ms, min_read_interval)) {
|
|
last_read_time_ms = millis();
|
|
|
|
uint32_t raw = 0;
|
|
float scaled = 0;
|
|
|
|
adcEnable();
|
|
#ifdef ARCH_ESP32 // ADC block for espressif platforms
|
|
raw = espAdcRead();
|
|
scaled = esp_adc_cal_raw_to_voltage(raw, adc_characs);
|
|
scaled *= operativeAdcMultiplier;
|
|
#else // block for all other platforms
|
|
for (uint32_t i = 0; i < BATTERY_SENSE_SAMPLES; i++) {
|
|
raw += analogRead(BATTERY_PIN);
|
|
}
|
|
raw = raw / BATTERY_SENSE_SAMPLES;
|
|
scaled = operativeAdcMultiplier * ((1000 * AREF_VOLTAGE) / pow(2, BATTERY_SENSE_RESOLUTION_BITS)) * raw;
|
|
#endif
|
|
adcDisable();
|
|
|
|
if (!initial_read_done) {
|
|
// Flush the smoothing filter with an ADC reading, if the reading is plausibly correct
|
|
if (scaled > last_read_value)
|
|
last_read_value = scaled;
|
|
initial_read_done = true;
|
|
} else {
|
|
// Already initialized - filter this reading
|
|
last_read_value += (scaled - last_read_value) * 0.5; // Virtual LPF
|
|
}
|
|
|
|
// LOG_DEBUG("battery gpio %d raw val=%u scaled=%u filtered=%u", BATTERY_PIN, raw, (uint32_t)(scaled), (uint32_t)
|
|
// (last_read_value));
|
|
}
|
|
return last_read_value;
|
|
#endif // BATTERY_PIN
|
|
return 0;
|
|
}
|
|
|
|
#if defined(ARCH_ESP32) && !defined(HAS_PMU) && defined(BATTERY_PIN)
|
|
/**
|
|
* ESP32 specific function for getting calibrated ADC reads
|
|
*/
|
|
uint32_t espAdcRead()
|
|
{
|
|
|
|
uint32_t raw = 0;
|
|
uint8_t raw_c = 0; // raw reading counter
|
|
|
|
#ifndef BAT_MEASURE_ADC_UNIT // ADC1
|
|
for (int i = 0; i < BATTERY_SENSE_SAMPLES; i++) {
|
|
int val_ = adc1_get_raw(adc_channel);
|
|
if (val_ >= 0) { // save only valid readings
|
|
raw += val_;
|
|
raw_c++;
|
|
}
|
|
// delayMicroseconds(100);
|
|
}
|
|
#else // ADC2
|
|
#ifdef CONFIG_IDF_TARGET_ESP32S3 // ESP32S3
|
|
// ADC2 wifi bug workaround not required, breaks compile
|
|
// On ESP32S3, ADC2 can take turns with Wifi (?)
|
|
|
|
int32_t adc_buf;
|
|
esp_err_t read_result;
|
|
|
|
// Multiple samples
|
|
for (int i = 0; i < BATTERY_SENSE_SAMPLES; i++) {
|
|
adc_buf = 0;
|
|
read_result = -1;
|
|
|
|
read_result = adc2_get_raw(adc_channel, ADC_WIDTH_BIT_12, &adc_buf);
|
|
if (read_result == ESP_OK) {
|
|
raw += adc_buf;
|
|
raw_c++; // Count valid samples
|
|
} else {
|
|
LOG_DEBUG("An attempt to sample ADC2 failed");
|
|
}
|
|
}
|
|
|
|
#else // Other ESP32
|
|
int32_t adc_buf = 0;
|
|
for (int i = 0; i < BATTERY_SENSE_SAMPLES; i++) {
|
|
// 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);
|
|
adc2_get_raw(adc_channel, ADC_WIDTH_BIT_12, &adc_buf);
|
|
raw += adc_buf;
|
|
raw_c++;
|
|
}
|
|
#endif // BAT_MEASURE_ADC_UNIT
|
|
|
|
#endif // End BAT_MEASURE_ADC_UNIT
|
|
return (raw / (raw_c < 1 ? 1 : raw_c));
|
|
}
|
|
#endif
|
|
|
|
/**
|
|
* return true if there is a battery installed in this unit
|
|
*/
|
|
// if we have a integrated device with a battery, we can assume that the battery is always connected
|
|
#ifdef BATTERY_IMMUTABLE
|
|
virtual bool isBatteryConnect() override { return true; }
|
|
#else
|
|
virtual bool isBatteryConnect() override { return getBatteryPercent() != -1; }
|
|
#endif
|
|
|
|
/// If we see a battery voltage higher than physics allows - assume charger is pumping
|
|
/// in power
|
|
/// On some boards we don't have the power management chip (like AXPxxxx)
|
|
/// so we use EXT_PWR_DETECT GPIO pin to detect external power source
|
|
virtual bool isVbusIn() override
|
|
{
|
|
#ifdef EXT_PWR_DETECT
|
|
#ifdef HELTEC_CAPSULE_SENSOR_V3
|
|
// if external powered that pin will be pulled down
|
|
if (digitalRead(EXT_PWR_DETECT) == LOW) {
|
|
return true;
|
|
}
|
|
// if it's not LOW - check the battery
|
|
#else
|
|
// if external powered that pin will be pulled up
|
|
if (digitalRead(EXT_PWR_DETECT) == HIGH) {
|
|
return true;
|
|
}
|
|
// if it's not HIGH - check the battery
|
|
#endif
|
|
#endif
|
|
return getBattVoltage() > chargingVolt;
|
|
}
|
|
|
|
/// Assume charging if we have a battery and external power is connected.
|
|
/// we can't be smart enough to say 'full'?
|
|
virtual bool isCharging() override
|
|
{
|
|
#if defined(HAS_RAKPROT) && !defined(ARCH_PORTDUINO) && !defined(HAS_PMU)
|
|
if (hasRAK()) {
|
|
return (rak9154Sensor.isCharging()) ? OptTrue : OptFalse;
|
|
}
|
|
#endif
|
|
#ifdef EXT_CHRG_DETECT
|
|
return digitalRead(EXT_CHRG_DETECT) == ext_chrg_detect_value;
|
|
#else
|
|
#if HAS_TELEMETRY && !MESHTASTIC_EXCLUDE_ENVIRONMENTAL_SENSOR && !defined(ARCH_PORTDUINO) && !defined(ARCH_STM32WL) && \
|
|
!defined(DISABLE_INA_CHARGING_DETECTION)
|
|
if (hasINA()) {
|
|
// get current flow from INA sensor - negative value means power flowing into the battery
|
|
// default assuming BATTERY+ <--> INA_VIN+ <--> SHUNT RESISTOR <--> INA_VIN- <--> LOAD
|
|
LOG_DEBUG("Using INA on I2C addr 0x%x for charging detection", config.power.device_battery_ina_address);
|
|
#if defined(INA_CHARGING_DETECTION_INVERT)
|
|
return getINACurrent() > 0;
|
|
#else
|
|
return getINACurrent() < 0;
|
|
#endif
|
|
}
|
|
return isBatteryConnect() && isVbusIn();
|
|
#endif
|
|
#endif
|
|
}
|
|
|
|
private:
|
|
/// If we see a battery voltage higher than physics allows - assume charger is pumping
|
|
/// in power
|
|
|
|
/// For heltecs with no battery connected, the measured voltage is 2204, so
|
|
// need to be higher than that, in this case is 2500mV (3000-500)
|
|
const uint16_t OCV[NUM_OCV_POINTS] = {OCV_ARRAY};
|
|
const float chargingVolt = (OCV[0] + 10) * NUM_CELLS;
|
|
const float noBatVolt = (OCV[NUM_OCV_POINTS - 1] - 500) * NUM_CELLS;
|
|
// Start value from minimum voltage for the filter to not start from 0
|
|
// that could trigger some events.
|
|
// This value is over-written by the first ADC reading, it the voltage seems reasonable.
|
|
bool initial_read_done = false;
|
|
float last_read_value = (OCV[NUM_OCV_POINTS - 1] * NUM_CELLS);
|
|
uint32_t last_read_time_ms = 0;
|
|
|
|
#if defined(HAS_RAKPROT)
|
|
|
|
uint16_t getRAKVoltage() { return rak9154Sensor.getBusVoltageMv(); }
|
|
|
|
bool hasRAK()
|
|
{
|
|
if (!rak9154Sensor.isInitialized())
|
|
return rak9154Sensor.runOnce() > 0;
|
|
return rak9154Sensor.isRunning();
|
|
}
|
|
#endif
|
|
|
|
#if HAS_TELEMETRY && !MESHTASTIC_EXCLUDE_ENVIRONMENTAL_SENSOR && !defined(ARCH_PORTDUINO) && !defined(ARCH_STM32WL)
|
|
uint16_t getINAVoltage()
|
|
{
|
|
if (nodeTelemetrySensorsMap[meshtastic_TelemetrySensorType_INA219].first == config.power.device_battery_ina_address) {
|
|
return ina219Sensor.getBusVoltageMv();
|
|
} else if (nodeTelemetrySensorsMap[meshtastic_TelemetrySensorType_INA226].first ==
|
|
config.power.device_battery_ina_address) {
|
|
return ina226Sensor.getBusVoltageMv();
|
|
} else if (nodeTelemetrySensorsMap[meshtastic_TelemetrySensorType_INA260].first ==
|
|
config.power.device_battery_ina_address) {
|
|
return ina260Sensor.getBusVoltageMv();
|
|
} else if (nodeTelemetrySensorsMap[meshtastic_TelemetrySensorType_INA3221].first ==
|
|
config.power.device_battery_ina_address) {
|
|
return ina3221Sensor.getBusVoltageMv();
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int16_t getINACurrent()
|
|
{
|
|
if (nodeTelemetrySensorsMap[meshtastic_TelemetrySensorType_INA219].first == config.power.device_battery_ina_address) {
|
|
return ina219Sensor.getCurrentMa();
|
|
} else if (nodeTelemetrySensorsMap[meshtastic_TelemetrySensorType_INA226].first ==
|
|
config.power.device_battery_ina_address) {
|
|
return ina226Sensor.getCurrentMa();
|
|
} else if (nodeTelemetrySensorsMap[meshtastic_TelemetrySensorType_INA3221].first ==
|
|
config.power.device_battery_ina_address) {
|
|
return ina3221Sensor.getCurrentMa();
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
bool hasINA()
|
|
{
|
|
if (!config.power.device_battery_ina_address) {
|
|
return false;
|
|
}
|
|
if (nodeTelemetrySensorsMap[meshtastic_TelemetrySensorType_INA219].first == config.power.device_battery_ina_address) {
|
|
if (!ina219Sensor.isInitialized())
|
|
return ina219Sensor.runOnce() > 0;
|
|
return ina219Sensor.isRunning();
|
|
} else if (nodeTelemetrySensorsMap[meshtastic_TelemetrySensorType_INA226].first ==
|
|
config.power.device_battery_ina_address) {
|
|
if (!ina226Sensor.isInitialized())
|
|
return ina226Sensor.runOnce() > 0;
|
|
} else if (nodeTelemetrySensorsMap[meshtastic_TelemetrySensorType_INA260].first ==
|
|
config.power.device_battery_ina_address) {
|
|
if (!ina260Sensor.isInitialized())
|
|
return ina260Sensor.runOnce() > 0;
|
|
return ina260Sensor.isRunning();
|
|
} else if (nodeTelemetrySensorsMap[meshtastic_TelemetrySensorType_INA3221].first ==
|
|
config.power.device_battery_ina_address) {
|
|
if (!ina3221Sensor.isInitialized())
|
|
return ina3221Sensor.runOnce() > 0;
|
|
return ina3221Sensor.isRunning();
|
|
}
|
|
return false;
|
|
}
|
|
#endif
|
|
};
|
|
|
|
static AnalogBatteryLevel analogLevel;
|
|
|
|
Power::Power() : OSThread("Power")
|
|
{
|
|
statusHandler = {};
|
|
low_voltage_counter = 0;
|
|
#ifdef DEBUG_HEAP
|
|
lastheap = memGet.getFreeHeap();
|
|
#endif
|
|
}
|
|
|
|
bool Power::analogInit()
|
|
{
|
|
#ifdef EXT_PWR_DETECT
|
|
#ifdef HELTEC_CAPSULE_SENSOR_V3
|
|
pinMode(EXT_PWR_DETECT, INPUT_PULLUP);
|
|
#else
|
|
pinMode(EXT_PWR_DETECT, INPUT);
|
|
#endif
|
|
#endif
|
|
#ifdef EXT_CHRG_DETECT
|
|
pinMode(EXT_CHRG_DETECT, ext_chrg_detect_mode);
|
|
#endif
|
|
|
|
#ifdef BATTERY_PIN
|
|
LOG_DEBUG("Use analog input %d for battery level", BATTERY_PIN);
|
|
|
|
// disable any internal pullups
|
|
pinMode(BATTERY_PIN, INPUT);
|
|
|
|
#ifndef BATTERY_SENSE_RESOLUTION_BITS
|
|
#define BATTERY_SENSE_RESOLUTION_BITS 10
|
|
#endif
|
|
|
|
#ifdef ARCH_ESP32 // ESP32 needs special analog stuff
|
|
|
|
#ifndef ADC_WIDTH // max resolution by default
|
|
static const adc_bits_width_t width = ADC_WIDTH_BIT_12;
|
|
#else
|
|
static const adc_bits_width_t width = ADC_WIDTH;
|
|
#endif
|
|
#ifndef BAT_MEASURE_ADC_UNIT // ADC1
|
|
adc1_config_width(width);
|
|
adc1_config_channel_atten(adc_channel, atten);
|
|
#else // ADC2
|
|
adc2_config_channel_atten(adc_channel, atten);
|
|
#ifndef CONFIG_IDF_TARGET_ESP32S3
|
|
// ADC2 wifi bug workaround
|
|
// Not required with ESP32S3, breaks compile
|
|
RTC_reg_b = READ_PERI_REG(SENS_SAR_READ_CTRL2_REG);
|
|
#endif
|
|
#endif
|
|
// calibrate ADC
|
|
esp_adc_cal_value_t val_type = esp_adc_cal_characterize(unit, atten, width, DEFAULT_VREF, adc_characs);
|
|
// show ADC characterization base
|
|
if (val_type == ESP_ADC_CAL_VAL_EFUSE_TP) {
|
|
LOG_INFO("ADC config based on Two Point values stored in eFuse");
|
|
} else if (val_type == ESP_ADC_CAL_VAL_EFUSE_VREF) {
|
|
LOG_INFO("ADC config based on reference voltage stored in eFuse");
|
|
}
|
|
#ifdef CONFIG_IDF_TARGET_ESP32S3
|
|
// ESP32S3
|
|
else if (val_type == ESP_ADC_CAL_VAL_EFUSE_TP_FIT) {
|
|
LOG_INFO("ADC config based on Two Point values and fitting curve coefficients stored in eFuse");
|
|
}
|
|
#endif
|
|
else {
|
|
LOG_INFO("ADC config based on default reference voltage");
|
|
}
|
|
#endif // ARCH_ESP32
|
|
|
|
#ifdef ARCH_NRF52
|
|
#ifdef VBAT_AR_INTERNAL
|
|
analogReference(VBAT_AR_INTERNAL);
|
|
#else
|
|
analogReference(AR_INTERNAL); // 3.6V
|
|
#endif
|
|
#endif // ARCH_NRF52
|
|
|
|
#ifndef ARCH_ESP32
|
|
analogReadResolution(BATTERY_SENSE_RESOLUTION_BITS);
|
|
#endif
|
|
|
|
batteryLevel = &analogLevel;
|
|
return true;
|
|
#else
|
|
return false;
|
|
#endif
|
|
}
|
|
|
|
/**
|
|
* Initializes the Power class.
|
|
*
|
|
* @return true if the setup was successful, false otherwise.
|
|
*/
|
|
bool Power::setup()
|
|
{
|
|
// initialise one power sensor (only)
|
|
bool found = axpChipInit();
|
|
if (!found)
|
|
found = lipoInit();
|
|
if (!found)
|
|
found = analogInit();
|
|
|
|
#ifdef NRF_APM
|
|
found = true;
|
|
#endif
|
|
|
|
enabled = found;
|
|
low_voltage_counter = 0;
|
|
|
|
return found;
|
|
}
|
|
|
|
void Power::shutdown()
|
|
{
|
|
LOG_INFO("Shutting down");
|
|
|
|
#if defined(ARCH_NRF52) || defined(ARCH_ESP32) || defined(ARCH_RP2040)
|
|
#ifdef PIN_LED1
|
|
ledOff(PIN_LED1);
|
|
#endif
|
|
#ifdef PIN_LED2
|
|
ledOff(PIN_LED2);
|
|
#endif
|
|
#ifdef PIN_LED3
|
|
ledOff(PIN_LED3);
|
|
#endif
|
|
doDeepSleep(DELAY_FOREVER, false, false);
|
|
#endif
|
|
}
|
|
|
|
/// Reads power status to powerStatus singleton.
|
|
//
|
|
// TODO(girts): move this and other axp stuff to power.h/power.cpp.
|
|
void Power::readPowerStatus()
|
|
{
|
|
int32_t batteryVoltageMv = -1; // Assume unknown
|
|
int8_t batteryChargePercent = -1;
|
|
OptionalBool usbPowered = OptUnknown;
|
|
OptionalBool hasBattery = OptUnknown; // These must be static because NRF_APM code doesn't run every time
|
|
OptionalBool isCharging = OptUnknown;
|
|
|
|
if (batteryLevel) {
|
|
hasBattery = batteryLevel->isBatteryConnect() ? OptTrue : OptFalse;
|
|
usbPowered = batteryLevel->isVbusIn() ? OptTrue : OptFalse;
|
|
isCharging = batteryLevel->isCharging() ? OptTrue : OptFalse;
|
|
if (hasBattery) {
|
|
batteryVoltageMv = batteryLevel->getBattVoltage();
|
|
// If the AXP192 returns a valid battery percentage, use it
|
|
if (batteryLevel->getBatteryPercent() >= 0) {
|
|
batteryChargePercent = batteryLevel->getBatteryPercent();
|
|
} else {
|
|
// If the AXP192 returns a percentage less than 0, the feature is either not supported or there is an error
|
|
// In that case, we compute an estimate of the charge percent based on open circuit voltage table defined
|
|
// in power.h
|
|
batteryChargePercent = clamp((int)(((batteryVoltageMv - (OCV[NUM_OCV_POINTS - 1] * NUM_CELLS)) * 1e2) /
|
|
((OCV[0] * NUM_CELLS) - (OCV[NUM_OCV_POINTS - 1] * NUM_CELLS))),
|
|
0, 100);
|
|
}
|
|
}
|
|
}
|
|
|
|
// FIXME: IMO we shouldn't be littering our code with all these ifdefs. Way better instead to make a Nrf52IsUsbPowered subclass
|
|
// (which shares a superclass with the BatteryLevel stuff)
|
|
// that just provides a few methods. But in the interest of fixing this bug I'm going to follow current
|
|
// practice.
|
|
#ifdef NRF_APM // Section of code detects USB power on the RAK4631 and updates the power states. Takes 20 seconds or so to detect
|
|
// changes.
|
|
|
|
nrfx_power_usb_state_t nrf_usb_state = nrfx_power_usbstatus_get();
|
|
// LOG_DEBUG("NRF Power %d", nrf_usb_state);
|
|
|
|
// If changed to DISCONNECTED
|
|
if (nrf_usb_state == NRFX_POWER_USB_STATE_DISCONNECTED)
|
|
isCharging = usbPowered = OptFalse;
|
|
// If changed to CONNECTED / READY
|
|
else
|
|
isCharging = usbPowered = OptTrue;
|
|
|
|
#endif
|
|
|
|
// Notify any status instances that are observing us
|
|
const PowerStatus powerStatus2 = PowerStatus(hasBattery, usbPowered, isCharging, batteryVoltageMv, batteryChargePercent);
|
|
LOG_DEBUG("Battery: usbPower=%d, isCharging=%d, batMv=%d, batPct=%d", powerStatus2.getHasUSB(), powerStatus2.getIsCharging(),
|
|
powerStatus2.getBatteryVoltageMv(), powerStatus2.getBatteryChargePercent());
|
|
newStatus.notifyObservers(&powerStatus2);
|
|
#ifdef DEBUG_HEAP
|
|
if (lastheap != memGet.getFreeHeap()) {
|
|
std::string threadlist = "Threads running:";
|
|
int running = 0;
|
|
for (int i = 0; i < MAX_THREADS; i++) {
|
|
auto thread = concurrency::mainController.get(i);
|
|
if ((thread != nullptr) && (thread->enabled)) {
|
|
threadlist += vformat(" %s", thread->ThreadName.c_str());
|
|
running++;
|
|
}
|
|
}
|
|
LOG_DEBUG(threadlist.c_str());
|
|
LOG_DEBUG("Heap status: %d/%d bytes free (%d), running %d/%d threads", memGet.getFreeHeap(), memGet.getHeapSize(),
|
|
memGet.getFreeHeap() - lastheap, running, concurrency::mainController.size(false));
|
|
lastheap = memGet.getFreeHeap();
|
|
}
|
|
#ifdef DEBUG_HEAP_MQTT
|
|
if (mqtt) {
|
|
// send MQTT-Packet with Heap-Size
|
|
uint8_t dmac[6];
|
|
getMacAddr(dmac); // Get our hardware ID
|
|
char mac[18];
|
|
sprintf(mac, "!%02x%02x%02x%02x", dmac[2], dmac[3], dmac[4], dmac[5]);
|
|
|
|
auto newHeap = memGet.getFreeHeap();
|
|
std::string heapTopic =
|
|
(*moduleConfig.mqtt.root ? moduleConfig.mqtt.root : "msh") + std::string("/2/heap/") + std::string(mac);
|
|
std::string heapString = std::to_string(newHeap);
|
|
mqtt->pubSub.publish(heapTopic.c_str(), heapString.c_str(), false);
|
|
auto wifiRSSI = WiFi.RSSI();
|
|
std::string wifiTopic =
|
|
(*moduleConfig.mqtt.root ? moduleConfig.mqtt.root : "msh") + std::string("/2/wifi/") + std::string(mac);
|
|
std::string wifiString = std::to_string(wifiRSSI);
|
|
mqtt->pubSub.publish(wifiTopic.c_str(), wifiString.c_str(), false);
|
|
}
|
|
#endif
|
|
|
|
#endif
|
|
|
|
// If we have a battery at all and it is less than 0%, force deep sleep if we have more than 10 low readings in
|
|
// a row. NOTE: min LiIon/LiPo voltage is 2.0 to 2.5V, current OCV min is set to 3100 that is large enough.
|
|
//
|
|
if (batteryLevel && powerStatus2.getHasBattery() && !powerStatus2.getHasUSB()) {
|
|
if (batteryLevel->getBattVoltage() < OCV[NUM_OCV_POINTS - 1]) {
|
|
low_voltage_counter++;
|
|
LOG_DEBUG("Low voltage counter: %d/10", low_voltage_counter);
|
|
if (low_voltage_counter > 10) {
|
|
#ifdef ARCH_NRF52
|
|
// We can't trigger deep sleep on NRF52, it's freezing the board
|
|
LOG_DEBUG("Low voltage detected, but not trigger deep sleep");
|
|
#else
|
|
LOG_INFO("Low voltage detected, trigger deep sleep");
|
|
powerFSM.trigger(EVENT_LOW_BATTERY);
|
|
#endif
|
|
}
|
|
} else {
|
|
low_voltage_counter = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
int32_t Power::runOnce()
|
|
{
|
|
readPowerStatus();
|
|
|
|
#ifdef HAS_PMU
|
|
// WE no longer use the IRQ line to wake the CPU (due to false wakes from sleep), but we do poll
|
|
// the IRQ status by reading the registers over I2C
|
|
if (PMU) {
|
|
|
|
PMU->getIrqStatus();
|
|
|
|
if (PMU->isVbusRemoveIrq()) {
|
|
LOG_INFO("USB unplugged");
|
|
powerFSM.trigger(EVENT_POWER_DISCONNECTED);
|
|
}
|
|
|
|
if (PMU->isVbusInsertIrq()) {
|
|
LOG_INFO("USB plugged In");
|
|
powerFSM.trigger(EVENT_POWER_CONNECTED);
|
|
}
|
|
|
|
/*
|
|
Other things we could check if we cared...
|
|
|
|
if (PMU->isBatChagerStartIrq()) {
|
|
LOG_DEBUG("Battery start charging");
|
|
}
|
|
if (PMU->isBatChagerDoneIrq()) {
|
|
LOG_DEBUG("Battery fully charged");
|
|
}
|
|
if (PMU->isBatInsertIrq()) {
|
|
LOG_DEBUG("Battery inserted");
|
|
}
|
|
if (PMU->isBatRemoveIrq()) {
|
|
LOG_DEBUG("Battery removed");
|
|
}
|
|
*/
|
|
#ifndef T_WATCH_S3 // FIXME - why is this triggering on the T-Watch S3?
|
|
if (PMU->isPekeyLongPressIrq()) {
|
|
LOG_DEBUG("PEK long button press");
|
|
screen->setOn(false);
|
|
}
|
|
#endif
|
|
|
|
PMU->clearIrqStatus();
|
|
}
|
|
#endif
|
|
// Only read once every 20 seconds once the power status for the app has been initialized
|
|
return (statusHandler && statusHandler->isInitialized()) ? (1000 * 20) : RUN_SAME;
|
|
}
|
|
|
|
/**
|
|
* Init the power manager chip
|
|
*
|
|
* axp192 power
|
|
DCDC1 0.7-3.5V @ 1200mA max -> OLED // If you turn this off you'll lose comms to the axp192 because the OLED and the
|
|
axp192 share the same i2c bus, instead use ssd1306 sleep mode DCDC2 -> unused DCDC3 0.7-3.5V @ 700mA max -> ESP32 (keep this
|
|
on!) LDO1 30mA -> charges GPS backup battery // charges the tiny J13 battery by the GPS to power the GPS ram (for a couple of
|
|
days), can not be turned off LDO2 200mA -> LORA LDO3 200mA -> GPS
|
|
*
|
|
*/
|
|
bool Power::axpChipInit()
|
|
{
|
|
|
|
#ifdef HAS_PMU
|
|
|
|
TwoWire *w = NULL;
|
|
|
|
// Use macro to distinguish which wire is used by PMU
|
|
#ifdef PMU_USE_WIRE1
|
|
w = &Wire1;
|
|
#else
|
|
w = &Wire;
|
|
#endif
|
|
|
|
/**
|
|
* It is not necessary to specify the wire pin,
|
|
* just input the wire, because the wire has been initialized in main.cpp
|
|
*/
|
|
if (!PMU) {
|
|
PMU = new XPowersAXP2101(*w);
|
|
if (!PMU->init()) {
|
|
LOG_WARN("No AXP2101 power management");
|
|
delete PMU;
|
|
PMU = NULL;
|
|
} else {
|
|
LOG_INFO("AXP2101 PMU init succeeded");
|
|
}
|
|
}
|
|
|
|
if (!PMU) {
|
|
PMU = new XPowersAXP192(*w);
|
|
if (!PMU->init()) {
|
|
LOG_WARN("No AXP192 power management");
|
|
delete PMU;
|
|
PMU = NULL;
|
|
} else {
|
|
LOG_INFO("AXP192 PMU init succeeded");
|
|
}
|
|
}
|
|
|
|
if (!PMU) {
|
|
/*
|
|
* In XPowersLib, if the XPowersAXPxxx object is released, Wire.end() will be called at the same time.
|
|
* In order not to affect other devices, if the initialization of the PMU fails, Wire needs to be re-initialized once,
|
|
* if there are multiple devices sharing the bus.
|
|
* * */
|
|
#ifndef PMU_USE_WIRE1
|
|
w->begin(I2C_SDA, I2C_SCL);
|
|
#endif
|
|
return false;
|
|
}
|
|
|
|
batteryLevel = PMU;
|
|
|
|
if (PMU->getChipModel() == XPOWERS_AXP192) {
|
|
|
|
// lora radio power channel
|
|
PMU->setPowerChannelVoltage(XPOWERS_LDO2, 3300);
|
|
PMU->enablePowerOutput(XPOWERS_LDO2);
|
|
|
|
// oled module power channel,
|
|
// disable it will cause abnormal communication between boot and AXP power supply,
|
|
// do not turn it off
|
|
PMU->setPowerChannelVoltage(XPOWERS_DCDC1, 3300);
|
|
// enable oled power
|
|
PMU->enablePowerOutput(XPOWERS_DCDC1);
|
|
|
|
// gnss module power channel - now turned on in setGpsPower
|
|
PMU->setPowerChannelVoltage(XPOWERS_LDO3, 3300);
|
|
// PMU->enablePowerOutput(XPOWERS_LDO3);
|
|
|
|
// protected oled power source
|
|
PMU->setProtectedChannel(XPOWERS_DCDC1);
|
|
// protected esp32 power source
|
|
PMU->setProtectedChannel(XPOWERS_DCDC3);
|
|
|
|
// disable not use channel
|
|
PMU->disablePowerOutput(XPOWERS_DCDC2);
|
|
|
|
// disable all axp chip interrupt
|
|
PMU->disableIRQ(XPOWERS_AXP192_ALL_IRQ);
|
|
|
|
// Set constant current charging current
|
|
PMU->setChargerConstantCurr(XPOWERS_AXP192_CHG_CUR_450MA);
|
|
|
|
// Set up the charging voltage
|
|
PMU->setChargeTargetVoltage(XPOWERS_AXP192_CHG_VOL_4V2);
|
|
} else if (PMU->getChipModel() == XPOWERS_AXP2101) {
|
|
|
|
/*The alternative version of T-Beam 1.1 differs from T-Beam V1.1 in that it uses an AXP2101 power chip*/
|
|
if (HW_VENDOR == meshtastic_HardwareModel_TBEAM) {
|
|
// Unuse power channel
|
|
PMU->disablePowerOutput(XPOWERS_DCDC2);
|
|
PMU->disablePowerOutput(XPOWERS_DCDC3);
|
|
PMU->disablePowerOutput(XPOWERS_DCDC4);
|
|
PMU->disablePowerOutput(XPOWERS_DCDC5);
|
|
PMU->disablePowerOutput(XPOWERS_ALDO1);
|
|
PMU->disablePowerOutput(XPOWERS_ALDO4);
|
|
PMU->disablePowerOutput(XPOWERS_BLDO1);
|
|
PMU->disablePowerOutput(XPOWERS_BLDO2);
|
|
PMU->disablePowerOutput(XPOWERS_DLDO1);
|
|
PMU->disablePowerOutput(XPOWERS_DLDO2);
|
|
|
|
// GNSS RTC PowerVDD 3300mV
|
|
PMU->setPowerChannelVoltage(XPOWERS_VBACKUP, 3300);
|
|
PMU->enablePowerOutput(XPOWERS_VBACKUP);
|
|
|
|
// ESP32 VDD 3300mV
|
|
// ! No need to set, automatically open , Don't close it
|
|
// PMU->setPowerChannelVoltage(XPOWERS_DCDC1, 3300);
|
|
// PMU->setProtectedChannel(XPOWERS_DCDC1);
|
|
|
|
// LoRa VDD 3300mV
|
|
PMU->setPowerChannelVoltage(XPOWERS_ALDO2, 3300);
|
|
PMU->enablePowerOutput(XPOWERS_ALDO2);
|
|
|
|
// GNSS VDD 3300mV
|
|
PMU->setPowerChannelVoltage(XPOWERS_ALDO3, 3300);
|
|
PMU->enablePowerOutput(XPOWERS_ALDO3);
|
|
} else if (HW_VENDOR == meshtastic_HardwareModel_LILYGO_TBEAM_S3_CORE ||
|
|
HW_VENDOR == meshtastic_HardwareModel_T_WATCH_S3) {
|
|
// t-beam s3 core
|
|
/**
|
|
* gnss module power channel
|
|
* The default ALDO4 is off, you need to turn on the GNSS power first, otherwise it will be invalid during
|
|
* initialization
|
|
*/
|
|
PMU->setPowerChannelVoltage(XPOWERS_ALDO4, 3300);
|
|
PMU->enablePowerOutput(XPOWERS_ALDO4);
|
|
|
|
// lora radio power channel
|
|
PMU->setPowerChannelVoltage(XPOWERS_ALDO3, 3300);
|
|
PMU->enablePowerOutput(XPOWERS_ALDO3);
|
|
|
|
// m.2 interface
|
|
PMU->setPowerChannelVoltage(XPOWERS_DCDC3, 3300);
|
|
PMU->enablePowerOutput(XPOWERS_DCDC3);
|
|
|
|
/**
|
|
* ALDO2 cannot be turned off.
|
|
* It is a necessary condition for sensor communication.
|
|
* It must be turned on to properly access the sensor and screen
|
|
* It is also responsible for the power supply of PCF8563
|
|
*/
|
|
PMU->setPowerChannelVoltage(XPOWERS_ALDO2, 3300);
|
|
PMU->enablePowerOutput(XPOWERS_ALDO2);
|
|
|
|
// 6-axis , magnetometer ,bme280 , oled screen power channel
|
|
PMU->setPowerChannelVoltage(XPOWERS_ALDO1, 3300);
|
|
PMU->enablePowerOutput(XPOWERS_ALDO1);
|
|
|
|
// sdcard power channel
|
|
PMU->setPowerChannelVoltage(XPOWERS_BLDO1, 3300);
|
|
PMU->enablePowerOutput(XPOWERS_BLDO1);
|
|
|
|
#ifdef T_WATCH_S3
|
|
// DRV2605 power channel
|
|
PMU->setPowerChannelVoltage(XPOWERS_BLDO2, 3300);
|
|
PMU->enablePowerOutput(XPOWERS_BLDO2);
|
|
#endif
|
|
|
|
// PMU->setPowerChannelVoltage(XPOWERS_DCDC4, 3300);
|
|
// PMU->enablePowerOutput(XPOWERS_DCDC4);
|
|
|
|
// not use channel
|
|
PMU->disablePowerOutput(XPOWERS_DCDC2); // not elicited
|
|
PMU->disablePowerOutput(XPOWERS_DCDC5); // not elicited
|
|
PMU->disablePowerOutput(XPOWERS_DLDO1); // Invalid power channel, it does not exist
|
|
PMU->disablePowerOutput(XPOWERS_DLDO2); // Invalid power channel, it does not exist
|
|
PMU->disablePowerOutput(XPOWERS_VBACKUP);
|
|
}
|
|
|
|
// disable all axp chip interrupt
|
|
PMU->disableIRQ(XPOWERS_AXP2101_ALL_IRQ);
|
|
|
|
// Set the constant current charging current of AXP2101, temporarily use 500mA by default
|
|
PMU->setChargerConstantCurr(XPOWERS_AXP2101_CHG_CUR_500MA);
|
|
|
|
// Set up the charging voltage
|
|
PMU->setChargeTargetVoltage(XPOWERS_AXP2101_CHG_VOL_4V2);
|
|
}
|
|
|
|
PMU->clearIrqStatus();
|
|
|
|
// TBeam1.1 /T-Beam S3-Core has no external TS detection,
|
|
// it needs to be disabled, otherwise it will cause abnormal charging
|
|
PMU->disableTSPinMeasure();
|
|
|
|
// PMU->enableSystemVoltageMeasure();
|
|
PMU->enableVbusVoltageMeasure();
|
|
PMU->enableBattVoltageMeasure();
|
|
|
|
if (PMU->isChannelAvailable(XPOWERS_DCDC1)) {
|
|
LOG_DEBUG("DC1 : %s Voltage:%u mV ", PMU->isPowerChannelEnable(XPOWERS_DCDC1) ? "+" : "-",
|
|
PMU->getPowerChannelVoltage(XPOWERS_DCDC1));
|
|
}
|
|
if (PMU->isChannelAvailable(XPOWERS_DCDC2)) {
|
|
LOG_DEBUG("DC2 : %s Voltage:%u mV ", PMU->isPowerChannelEnable(XPOWERS_DCDC2) ? "+" : "-",
|
|
PMU->getPowerChannelVoltage(XPOWERS_DCDC2));
|
|
}
|
|
if (PMU->isChannelAvailable(XPOWERS_DCDC3)) {
|
|
LOG_DEBUG("DC3 : %s Voltage:%u mV ", PMU->isPowerChannelEnable(XPOWERS_DCDC3) ? "+" : "-",
|
|
PMU->getPowerChannelVoltage(XPOWERS_DCDC3));
|
|
}
|
|
if (PMU->isChannelAvailable(XPOWERS_DCDC4)) {
|
|
LOG_DEBUG("DC4 : %s Voltage:%u mV ", PMU->isPowerChannelEnable(XPOWERS_DCDC4) ? "+" : "-",
|
|
PMU->getPowerChannelVoltage(XPOWERS_DCDC4));
|
|
}
|
|
if (PMU->isChannelAvailable(XPOWERS_LDO2)) {
|
|
LOG_DEBUG("LDO2 : %s Voltage:%u mV ", PMU->isPowerChannelEnable(XPOWERS_LDO2) ? "+" : "-",
|
|
PMU->getPowerChannelVoltage(XPOWERS_LDO2));
|
|
}
|
|
if (PMU->isChannelAvailable(XPOWERS_LDO3)) {
|
|
LOG_DEBUG("LDO3 : %s Voltage:%u mV ", PMU->isPowerChannelEnable(XPOWERS_LDO3) ? "+" : "-",
|
|
PMU->getPowerChannelVoltage(XPOWERS_LDO3));
|
|
}
|
|
if (PMU->isChannelAvailable(XPOWERS_ALDO1)) {
|
|
LOG_DEBUG("ALDO1: %s Voltage:%u mV ", PMU->isPowerChannelEnable(XPOWERS_ALDO1) ? "+" : "-",
|
|
PMU->getPowerChannelVoltage(XPOWERS_ALDO1));
|
|
}
|
|
if (PMU->isChannelAvailable(XPOWERS_ALDO2)) {
|
|
LOG_DEBUG("ALDO2: %s Voltage:%u mV ", PMU->isPowerChannelEnable(XPOWERS_ALDO2) ? "+" : "-",
|
|
PMU->getPowerChannelVoltage(XPOWERS_ALDO2));
|
|
}
|
|
if (PMU->isChannelAvailable(XPOWERS_ALDO3)) {
|
|
LOG_DEBUG("ALDO3: %s Voltage:%u mV ", PMU->isPowerChannelEnable(XPOWERS_ALDO3) ? "+" : "-",
|
|
PMU->getPowerChannelVoltage(XPOWERS_ALDO3));
|
|
}
|
|
if (PMU->isChannelAvailable(XPOWERS_ALDO4)) {
|
|
LOG_DEBUG("ALDO4: %s Voltage:%u mV ", PMU->isPowerChannelEnable(XPOWERS_ALDO4) ? "+" : "-",
|
|
PMU->getPowerChannelVoltage(XPOWERS_ALDO4));
|
|
}
|
|
if (PMU->isChannelAvailable(XPOWERS_BLDO1)) {
|
|
LOG_DEBUG("BLDO1: %s Voltage:%u mV ", PMU->isPowerChannelEnable(XPOWERS_BLDO1) ? "+" : "-",
|
|
PMU->getPowerChannelVoltage(XPOWERS_BLDO1));
|
|
}
|
|
if (PMU->isChannelAvailable(XPOWERS_BLDO2)) {
|
|
LOG_DEBUG("BLDO2: %s Voltage:%u mV ", PMU->isPowerChannelEnable(XPOWERS_BLDO2) ? "+" : "-",
|
|
PMU->getPowerChannelVoltage(XPOWERS_BLDO2));
|
|
}
|
|
|
|
// We can safely ignore this approach for most (or all) boards because MCU turned off
|
|
// earlier than battery discharged to 2.6V.
|
|
//
|
|
// Unfortanly for now we can't use this killswitch for RAK4630-based boards because they have a bug with
|
|
// battery voltage measurement. Probably it sometimes drops to low values.
|
|
#ifndef RAK4630
|
|
// Set PMU shutdown voltage at 2.6V to maximize battery utilization
|
|
PMU->setSysPowerDownVoltage(2600);
|
|
#endif
|
|
|
|
#ifdef PMU_IRQ
|
|
uint64_t pmuIrqMask = 0;
|
|
|
|
if (PMU->getChipModel() == XPOWERS_AXP192) {
|
|
pmuIrqMask = XPOWERS_AXP192_VBUS_INSERT_IRQ | XPOWERS_AXP192_BAT_INSERT_IRQ | XPOWERS_AXP192_PKEY_SHORT_IRQ;
|
|
} else if (PMU->getChipModel() == XPOWERS_AXP2101) {
|
|
pmuIrqMask = XPOWERS_AXP2101_VBUS_INSERT_IRQ | XPOWERS_AXP2101_BAT_INSERT_IRQ | XPOWERS_AXP2101_PKEY_SHORT_IRQ;
|
|
}
|
|
|
|
pinMode(PMU_IRQ, INPUT);
|
|
attachInterrupt(
|
|
PMU_IRQ, [] { pmu_irq = true; }, FALLING);
|
|
|
|
// we do not look for AXPXXX_CHARGING_FINISHED_IRQ & AXPXXX_CHARGING_IRQ because it occurs repeatedly while there is
|
|
// no battery also it could cause inadvertent waking from light sleep just because the battery filled
|
|
// we don't look for AXPXXX_BATT_REMOVED_IRQ because it occurs repeatedly while no battery installed
|
|
// we don't look at AXPXXX_VBUS_REMOVED_IRQ because we don't have anything hooked to vbus
|
|
PMU->enableIRQ(pmuIrqMask);
|
|
|
|
PMU->clearIrqStatus();
|
|
#endif /*PMU_IRQ*/
|
|
|
|
readPowerStatus();
|
|
|
|
pmu_found = true;
|
|
|
|
return pmu_found;
|
|
|
|
#else
|
|
return false;
|
|
#endif
|
|
}
|
|
|
|
#if !MESHTASTIC_EXCLUDE_I2C && !defined(ARCH_PORTDUINO) && !defined(ARCH_STM32WL)
|
|
|
|
/**
|
|
* Wrapper class for an I2C MAX17048 Lipo battery sensor.
|
|
*/
|
|
class LipoBatteryLevel : public HasBatteryLevel
|
|
{
|
|
private:
|
|
MAX17048Singleton *max17048 = nullptr;
|
|
|
|
public:
|
|
/**
|
|
* Init the I2C MAX17048 Lipo battery level sensor
|
|
*/
|
|
bool runOnce()
|
|
{
|
|
if (max17048 == nullptr) {
|
|
max17048 = MAX17048Singleton::GetInstance();
|
|
}
|
|
|
|
// try to start if the sensor has been detected
|
|
if (nodeTelemetrySensorsMap[meshtastic_TelemetrySensorType_MAX17048].first != 0) {
|
|
return max17048->runOnce(nodeTelemetrySensorsMap[meshtastic_TelemetrySensorType_MAX17048].second);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* Battery state of charge, from 0 to 100 or -1 for unknown
|
|
*/
|
|
virtual int getBatteryPercent() override { return max17048->getBusBatteryPercent(); }
|
|
|
|
/**
|
|
* The raw voltage of the battery in millivolts, or NAN if unknown
|
|
*/
|
|
virtual uint16_t getBattVoltage() override { return max17048->getBusVoltageMv(); }
|
|
|
|
/**
|
|
* return true if there is a battery installed in this unit
|
|
*/
|
|
virtual bool isBatteryConnect() override { return max17048->isBatteryConnected(); }
|
|
|
|
/**
|
|
* return true if there is an external power source detected
|
|
*/
|
|
virtual bool isVbusIn() override { return max17048->isExternallyPowered(); }
|
|
|
|
/**
|
|
* return true if the battery is currently charging
|
|
*/
|
|
virtual bool isCharging() override { return max17048->isBatteryCharging(); }
|
|
};
|
|
|
|
LipoBatteryLevel lipoLevel;
|
|
|
|
/**
|
|
* Init the Lipo battery level sensor
|
|
*/
|
|
bool Power::lipoInit()
|
|
{
|
|
bool result = lipoLevel.runOnce();
|
|
LOG_DEBUG("Power::lipoInit lipo sensor is %s", result ? "ready" : "not ready yet");
|
|
if (!result)
|
|
return false;
|
|
batteryLevel = &lipoLevel;
|
|
return true;
|
|
}
|
|
|
|
#else
|
|
/**
|
|
* The Lipo battery level sensor is unavailable - default to AnalogBatteryLevel
|
|
*/
|
|
bool Power::lipoInit()
|
|
{
|
|
return false;
|
|
}
|
|
#endif
|