-
-
Notifications
You must be signed in to change notification settings - Fork 40.8k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Implement battery level interface (#24666)
Co-authored-by: Nick Brassel <[email protected]>
- Loading branch information
Showing
12 changed files
with
275 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,51 @@ | ||
# Battery Driver | ||
|
||
This driver provides support for sampling battery level. | ||
|
||
## Usage | ||
|
||
To use this driver, add the following to your `rules.mk`: | ||
|
||
```make | ||
BATTERY_DRIVER_REQUIRED = yes | ||
``` | ||
|
||
## Basic Configuration {#basic-configuration} | ||
|
||
Add the following to your `config.h`: | ||
|
||
|Define |Default |Description | | ||
|--------------------------|--------|--------------------------------------------------| | ||
|`BATTERY_SAMPLE_INTERVAL` |`30000` |The time between battery samples in milliseconds. | | ||
|
||
## Driver Configuration {#driver-configuration} | ||
|
||
Driver selection can be configured in `rules.mk` as `BATTERY_DRIVER`. Valid values are `adc` (default), `vendor`, or `custom`. See below for information on individual drivers. | ||
|
||
### ADC Driver {#adc-driver} | ||
|
||
This is the default battery driver. The default configuration assumes the battery is connected to a ADC capable pin through a voltage divider. | ||
|
||
```make | ||
BATTERY_DRIVER = adc | ||
``` | ||
|
||
The following `#define`s apply only to the `adc` driver: | ||
|
||
|Define |Default |Description | | ||
|-----------------------------|--------------|--------------------------------------------------------------| | ||
|`BATTERY_PIN` |*Not defined* |The GPIO pin connected to the voltage divider. | | ||
|`BATTERY_REF_VOLTAGE_MV` |`3300` |The ADC reverence voltage, in millivolts. | | ||
|`BATTERY_VOLTAGE_DIVIDER_R1` |`100000` |The voltage divider resistance, in kOhm. Set to 0 to disable. | | ||
|`BATTERY_VOLTAGE_DIVIDER_R1` |`100000` |The voltage divider resistance, in kOhm. Set to 0 to disable. | | ||
|`BATTERY_ADC_RESOLUTION` |`10` |The ADC resolution configured for the ADC Driver. | | ||
|
||
## Functions | ||
|
||
### `uint8_t battery_get_percent(void)` {#api-battery-get-percent} | ||
|
||
Sample battery level. | ||
|
||
#### Return Value {#api-battery-get-percent-return} | ||
|
||
The battery percentage, in the range 0-100. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,31 @@ | ||
// 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(); | ||
} | ||
|
||
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(); | ||
|
||
bat_timer = timer_read32(); | ||
} | ||
} | ||
|
||
uint8_t battery_get_percent(void) { | ||
return last_bat_level; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,34 @@ | ||
// 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); | ||
|
||
/** \} */ |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,55 @@ | ||
// Copyright 2025 QMK | ||
// SPDX-License-Identifier: GPL-2.0-or-later | ||
|
||
#include "analog.h" | ||
#include "gpio.h" | ||
|
||
#ifndef BATTERY_PIN | ||
# error("BATTERY_PIN not configured!") | ||
#endif | ||
|
||
#ifndef BATTERY_REF_VOLTAGE_MV | ||
# define BATTERY_REF_VOLTAGE_MV 3300 | ||
#endif | ||
|
||
#ifndef BATTERY_VOLTAGE_DIVIDER_R1 | ||
# define BATTERY_VOLTAGE_DIVIDER_R1 100 | ||
#endif | ||
|
||
#ifndef BATTERY_VOLTAGE_DIVIDER_R2 | ||
# define BATTERY_VOLTAGE_DIVIDER_R2 100 | ||
#endif | ||
|
||
// TODO: infer from adc config? | ||
#ifndef BATTERY_ADC_RESOLUTION | ||
# define BATTERY_ADC_RESOLUTION 10 | ||
#endif | ||
|
||
void battery_driver_init(void) { | ||
gpio_set_pin_input(BATTERY_PIN); | ||
} | ||
|
||
uint16_t battery_driver_get_mv(void) { | ||
uint32_t raw = analogReadPin(BATTERY_PIN); | ||
|
||
uint32_t bat_mv = raw * BATTERY_REF_VOLTAGE_MV / (1 << BATTERY_ADC_RESOLUTION); | ||
|
||
#if BATTERY_VOLTAGE_DIVIDER_R1 > 0 && BATTERY_VOLTAGE_DIVIDER_R2 > 0 | ||
bat_mv = bat_mv * (BATTERY_VOLTAGE_DIVIDER_R1 + BATTERY_VOLTAGE_DIVIDER_R2) / BATTERY_VOLTAGE_DIVIDER_R2; | ||
#endif | ||
|
||
return bat_mv; | ||
} | ||
|
||
uint8_t battery_driver_sample_percent(void) { | ||
uint16_t bat_mv = battery_driver_get_mv(); | ||
|
||
// https://github.com/zmkfirmware/zmk/blob/3f7c9d7cc4f46617faad288421025ea2a6b0bd28/app/module/drivers/sensor/battery/battery_common.c#L33 | ||
if (bat_mv >= 4200) { | ||
return 100; | ||
} else if (bat_mv <= 3450) { | ||
return 0; | ||
} | ||
|
||
return bat_mv * 2 / 15 - 459; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,29 @@ | ||
// Copyright 2025 QMK | ||
// SPDX-License-Identifier: GPL-2.0-or-later | ||
|
||
#pragma once | ||
|
||
#include <stdint.h> | ||
|
||
/** | ||
* \file | ||
* | ||
* \defgroup battery Battery Driver API | ||
* | ||
* \brief API to query battery status. | ||
* \{ | ||
*/ | ||
|
||
/** | ||
* \brief Initialize the battery driver. This function must be called only once, before any of the below functions can be called. | ||
*/ | ||
void battery_driver_init(void); | ||
|
||
/** | ||
* \brief Sample battery level. | ||
* | ||
* \return The battery percentage, in the range 0-100. | ||
*/ | ||
uint8_t battery_driver_sample_percent(void); | ||
|
||
/** \} */ |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,6 @@ | ||
// Copyright 2024 QMK | ||
// SPDX-License-Identifier: GPL-2.0-or-later | ||
|
||
#pragma once | ||
|
||
#define BATTERY_PIN ADC_PIN |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,28 @@ | ||
// Copyright 2024 QMK | ||
// SPDX-License-Identifier: GPL-2.0-or-later | ||
|
||
#include QMK_KEYBOARD_H | ||
#include "battery.h" | ||
|
||
const uint16_t PROGMEM keymaps[][MATRIX_ROWS][MATRIX_COLS] = { | ||
LAYOUT_ortho_1x1(KC_A) | ||
}; | ||
|
||
void keyboard_post_init_user(void) { | ||
// Customise these values to desired behaviour | ||
debug_enable=true; | ||
// debug_matrix=false; | ||
// debug_keyboard=true; | ||
// debug_mouse=false; | ||
|
||
battery_init(); | ||
} | ||
|
||
void housekeeping_task_user(void) { | ||
static uint32_t last = 0; | ||
if (timer_elapsed32(last) > 2000) { | ||
uprintf("Bat: %d!\n", battery_get_percent()); | ||
|
||
last = timer_read32(); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,7 @@ | ||
{ | ||
"config": { | ||
"features": { | ||
"console": true | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
BATTERY_DRIVER_REQUIRED = yes |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters