Refactor battery driver (#25550)

This commit is contained in:
Joel Challis 2025-08-17 01:14:48 +01:00 committed by GitHub
parent f29d8117bf
commit cc696a2ae8
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
26 changed files with 287 additions and 73 deletions

41
quantum/battery/battery.c Normal file
View file

@ -0,0 +1,41 @@
// Copyright 2025 QMK
// SPDX-License-Identifier: GPL-2.0-or-later
#include "battery_driver.h"
#include "battery.h"
#include "timer.h"
#ifndef BATTERY_SAMPLE_INTERVAL
# define BATTERY_SAMPLE_INTERVAL 30000
#endif
static uint8_t last_bat_level = 100;
void battery_init(void) {
battery_driver_init();
last_bat_level = battery_driver_sample_percent();
}
__attribute__((weak)) void battery_percent_changed_user(uint8_t level) {}
__attribute__((weak)) void battery_percent_changed_kb(uint8_t level) {}
static void handle_percent_changed(void) {
battery_percent_changed_user(last_bat_level);
battery_percent_changed_kb(last_bat_level);
}
void battery_task(void) {
static uint32_t bat_timer = 0;
if (timer_elapsed32(bat_timer) > BATTERY_SAMPLE_INTERVAL) {
last_bat_level = battery_driver_sample_percent();
handle_percent_changed();
bat_timer = timer_read32();
}
}
uint8_t battery_get_percent(void) {
return last_bat_level;
}

46
quantum/battery/battery.h Normal file
View file

@ -0,0 +1,46 @@
// Copyright 2025 QMK
// SPDX-License-Identifier: GPL-2.0-or-later
#pragma once
#include <stdint.h>
/**
* \file
*
* \defgroup battery Battery API
*
* \brief API to query battery status.
* \{
*/
/**
* \brief Initialize the battery driver.
*/
void battery_init(void);
/**
* \brief Perform housekeeping tasks.
*/
void battery_task(void);
/**
* \brief Sample battery level.
*
* \return The battery percentage, in the range 0-100.
*/
uint8_t battery_get_percent(void);
/**
* \brief user hook called when battery level changed.
*
*/
void battery_percent_changed_user(uint8_t level);
/**
* \brief keyboard hook called when battery level changed.
*
*/
void battery_percent_changed_kb(uint8_t level);
/** \} */

View file

@ -0,0 +1,97 @@
// Copyright 2025 QMK
// SPDX-License-Identifier: GPL-2.0-or-later
#include "gtest/gtest.h"
#include "gmock/gmock.h"
using testing::_;
class BatteryDriverMock {
public:
virtual ~BatteryDriverMock() {}
// mock methods
MOCK_METHOD0(battery_driver_init, void(void));
MOCK_METHOD0(battery_driver_sample_percent, uint8_t(void));
MOCK_METHOD1(battery_percent_changed_kb, void(uint8_t));
};
class BatteryTest : public ::testing::Test {
public:
BatteryTest() {
_batteryDriverMock.reset(new ::testing::NiceMock<BatteryDriverMock>());
}
virtual ~BatteryTest() {
_batteryDriverMock.reset();
}
static std::unique_ptr<BatteryDriverMock> _batteryDriverMock;
};
std::unique_ptr<BatteryDriverMock> BatteryTest::_batteryDriverMock;
extern "C" {
#include "quantum/battery/battery.h"
#include "timer.h"
void advance_time(uint32_t ms);
void battery_driver_init(void) {
if (BatteryTest::_batteryDriverMock) {
BatteryTest::_batteryDriverMock->battery_driver_init();
}
}
uint8_t battery_driver_sample_percent(void) {
if (BatteryTest::_batteryDriverMock) {
return BatteryTest::_batteryDriverMock->battery_driver_sample_percent();
}
return 255;
}
void battery_percent_changed_kb(uint8_t level) {
if (BatteryTest::_batteryDriverMock) {
BatteryTest::_batteryDriverMock->battery_percent_changed_kb(level);
}
}
}
TEST_F(BatteryTest, TestInit) {
// init driver and initial sample
EXPECT_CALL(*_batteryDriverMock, battery_driver_init()).Times(1);
EXPECT_CALL(*_batteryDriverMock, battery_driver_sample_percent()).Times(1);
battery_init();
}
TEST_F(BatteryTest, TestSampleCached) {
// sample before timeout
EXPECT_CALL(*_batteryDriverMock, battery_driver_sample_percent()).Times(0);
advance_time(1);
battery_task();
}
TEST_F(BatteryTest, TestSampleNotCached) {
// sample after timeout
EXPECT_CALL(*_batteryDriverMock, battery_driver_sample_percent()).Times(1);
advance_time(60000);
battery_task();
}
TEST_F(BatteryTest, TestGet) {
// sample does not directly sample
EXPECT_CALL(*_batteryDriverMock, battery_driver_sample_percent()).Times(0);
battery_get_percent();
}
TEST_F(BatteryTest, TestChanged) {
// callbacks on value changed
EXPECT_CALL(*_batteryDriverMock, battery_percent_changed_kb(_)).Times(1);
battery_task();
advance_time(60000);
battery_task();
}

View file

@ -0,0 +1,7 @@
VPATH += $(DRIVER_PATH)/battery
battery_SRC := \
$(PLATFORM_PATH)/timer.c \
$(PLATFORM_PATH)/$(PLATFORM_KEY)/timer.c \
$(QUANTUM_PATH)/battery/battery.c \
$(QUANTUM_PATH)/battery/tests/battery_tests.cpp \

View file

@ -0,0 +1,2 @@
TEST_LIST += \
battery \

View file

@ -122,7 +122,7 @@ along with this program. If not, see <http://www.gnu.org/licenses/>.
#ifdef SPLIT_KEYBOARD
# include "split_util.h"
#endif
#ifdef BATTERY_DRIVER
#ifdef BATTERY_ENABLE
# include "battery.h"
#endif
#ifdef BLUETOOTH_ENABLE
@ -532,7 +532,7 @@ void keyboard_init(void) {
// init after split init
pointing_device_init();
#endif
#ifdef BATTERY_DRIVER
#ifdef BATTERY_ENABLE
battery_init();
#endif
#ifdef BLUETOOTH_ENABLE
@ -779,7 +779,7 @@ void keyboard_task(void) {
joystick_task();
#endif
#ifdef BATTERY_DRIVER
#ifdef BATTERY_ENABLE
battery_task();
#endif

View file

@ -63,6 +63,10 @@
# include "bootmagic.h"
#endif
#ifdef BATTERY_ENABLE
# include "battery.h"
#endif
#ifdef DEFERRED_EXEC_ENABLE
# include "deferred_exec.h"
#endif