* [PATCH v3 0/1] platform/x86: add Acer battery control driver
@ 2026-05-10 18:50 Jelle van der Waa
2026-05-10 18:50 ` [PATCH v3 1/1] " Jelle van der Waa
0 siblings, 1 reply; 5+ messages in thread
From: Jelle van der Waa @ 2026-05-10 18:50 UTC (permalink / raw)
To: Hans de Goede, Ilpo Järvinen
Cc: Jelle van der Waa, platform-driver-x86, Frederik Harwath
This patch upstreams a part of the out of tree acer wmi battery
specifically the battery charge control and battery temperature. [1]
On my Acer Aspire A315-510P battery calibration did not work as expected
so for now this is left out.
Jelle van der Waa (1):
platform/x86: add Acer battery control driver
drivers/platform/x86/Kconfig | 12 +
drivers/platform/x86/Makefile | 1 +
drivers/platform/x86/acer-wmi-battery.c | 347 ++++++++++++++++++++++++
3 files changed, 360 insertions(+)
create mode 100644 drivers/platform/x86/acer-wmi-battery.c
--
2.54.0
^ permalink raw reply [flat|nested] 5+ messages in thread
* [PATCH v3 1/1] platform/x86: add Acer battery control driver
2026-05-10 18:50 [PATCH v3 0/1] platform/x86: add Acer battery control driver Jelle van der Waa
@ 2026-05-10 18:50 ` Jelle van der Waa
2026-05-11 11:54 ` Ilpo Järvinen
2026-05-11 16:09 ` Rong Zhang
0 siblings, 2 replies; 5+ messages in thread
From: Jelle van der Waa @ 2026-05-10 18:50 UTC (permalink / raw)
To: Hans de Goede, Ilpo Järvinen
Cc: Jelle van der Waa, platform-driver-x86, Frederik Harwath
Some Acer laptops can configure battery related features through Acer
Care Center on Windows. This driver uses the power supply extension to
set a battery charge limit and exposes the battery
temperature.
This driver is based on the existing acer-wmi-battery project on GitHub
and was tested on an Acer Aspire A315-510P.
Signed-off-by: Jelle van der Waa <jelle@vdwaa.nl>
---
v3:
- Add depends on DMI
- Rename CamelCase struct member names
- Simplify returning errors
- Add comma to non-terminating entries
- Simplified acer_wmi_battery_set_battery_health_control to acer_wmi_set_battery_health_control
- Use sizeof for acpi_object buffer
- Drop POWER_SUPPLY_EXTENSION macro
v2:
- Alphabetically sort linux headers
- Include headers for types / _packed
- Use cleanup.h instead of goto + label
- Add missing prefix for set_battery_health_control
- General code formatting fixes
- Remove HWMON dependency in Kconfig
- Use wmidev_evaluate_method()
- Accept oversized ACPI buffers
- Use DRIVER_NAME for battery extension name
- Set no_singleton = true
- Implement DMI matching to support laptops with only battery
temperature support.
---
drivers/platform/x86/Kconfig | 12 +
drivers/platform/x86/Makefile | 1 +
drivers/platform/x86/acer-wmi-battery.c | 347 ++++++++++++++++++++++++
3 files changed, 360 insertions(+)
create mode 100644 drivers/platform/x86/acer-wmi-battery.c
diff --git a/drivers/platform/x86/Kconfig b/drivers/platform/x86/Kconfig
index 2ffa4ecf65b0..2fc23cd2039f 100644
--- a/drivers/platform/x86/Kconfig
+++ b/drivers/platform/x86/Kconfig
@@ -188,6 +188,18 @@ config ACER_WMI
If you have an ACPI-WMI compatible Acer/ Wistron laptop, say Y or M
here.
+config ACER_WMI_BATTERY
+ tristate "Acer WMI Battery"
+ depends on ACPI_WMI
+ depends on ACPI_BATTERY
+ depends on DMI
+ help
+ This is a driver for Acer laptops with battery health control. It
+ adds charge limit control and battery temperature reporting.
+
+ If you have an ACPI-WMI Battery compatible Acer laptop, say Y or M
+ here.
+
source "drivers/platform/x86/amd/Kconfig"
config ADV_SWBUTTON
diff --git a/drivers/platform/x86/Makefile b/drivers/platform/x86/Makefile
index 872ac3842391..a877acd937cd 100644
--- a/drivers/platform/x86/Makefile
+++ b/drivers/platform/x86/Makefile
@@ -20,6 +20,7 @@ obj-$(CONFIG_BITLAND_MIFS_WMI) += bitland-mifs-wmi.o
obj-$(CONFIG_ACERHDF) += acerhdf.o
obj-$(CONFIG_ACER_WIRELESS) += acer-wireless.o
obj-$(CONFIG_ACER_WMI) += acer-wmi.o
+obj-$(CONFIG_ACER_WMI_BATTERY) += acer-wmi-battery.o
# AMD
obj-y += amd/
diff --git a/drivers/platform/x86/acer-wmi-battery.c b/drivers/platform/x86/acer-wmi-battery.c
new file mode 100644
index 000000000000..78005e7a7f55
--- /dev/null
+++ b/drivers/platform/x86/acer-wmi-battery.c
@@ -0,0 +1,347 @@
+// SPDX-License-Identifier: GPL-2.0-or-later
+/*
+ * acer-wmi-battery.c: Acer battery health control driver
+ *
+ * This is a driver for the WMI battery health control interface found
+ * on some Acer laptops. This interface allows to enable/disable a
+ * battery charge limit ("health mode") and exposes the battery temperature.
+ *
+ * Based on acer-wmi-battery https://github.com/frederik-h/acer-wmi-battery/
+ *
+ * Copyright (C) 2022-2025 Frederik Harwath <frederik@harwath.name>
+ */
+
+#include <linux/acpi.h>
+#include <linux/cleanup.h>
+#include <linux/compiler_attributes.h>
+#include <linux/dmi.h>
+#include <linux/init.h>
+#include <linux/kernel.h>
+#include <linux/limits.h>
+#include <linux/module.h>
+#include <linux/power_supply.h>
+#include <linux/types.h>
+#include <linux/unaligned.h>
+#include <linux/version.h>
+#include <linux/wmi.h>
+
+#include <acpi/battery.h>
+
+#define DRIVER_NAME "acer-wmi-battery"
+
+#define ACER_BATTERY_GUID "79772EC5-04B1-4BFD-843C-61E7F77B6CC9"
+
+/*
+ * The Acer OEM software seems to always use this battery index,
+ * so we emulate this behaviour to not confuse the underlying firmware.
+ *
+ * However this also means that we only fully support devices with a
+ * single battery for now.
+ */
+#define ACER_BATTERY_INDEX 0x1
+
+struct get_battery_health_control_status_input {
+ u8 battery_no;
+ u8 function_query;
+ u8 reserved[2];
+} __packed;
+
+struct get_battery_health_control_status_output {
+ u8 function_list;
+ u8 ret[2];
+ u8 function_status[5];
+} __packed;
+
+struct set_battery_health_control_input {
+ u8 battery_no;
+ u8 function_mask;
+ u8 function_status;
+ u8 reserved_in[5];
+} __packed;
+
+struct set_battery_health_control_output {
+ u8 ret;
+ u8 reserved_out;
+} __packed;
+
+enum battery_mode {
+ HEALTH_MODE = 1,
+ CALIBRATION_MODE = 2,
+};
+
+struct acer_wmi_battery_data {
+ struct acpi_battery_hook hook;
+ struct wmi_device *wdev;
+ const struct power_supply_ext *battery_ext;
+ struct {
+ bool health_mode : 1;
+ } features;
+};
+
+static int acer_wmi_battery_get_information(struct acer_wmi_battery_data *data,
+ u32 index, u32 battery, u32 *result)
+{
+ u32 args[2] = { index, battery };
+ struct acpi_buffer output = { ACPI_ALLOCATE_BUFFER, NULL };
+ struct acpi_buffer input = { sizeof(args), args };
+ int ret;
+
+ ret = wmidev_evaluate_method(data->wdev, 0, 19, &input, &output);
+ if (ACPI_FAILURE(ret))
+ return -EIO;
+
+ union acpi_object *obj __free(kfree) = output.pointer;
+ if (!obj || obj->type != ACPI_TYPE_BUFFER)
+ return -EIO;
+
+ if (obj->buffer.length < sizeof(u32)) {
+ dev_err(&data->wdev->dev, "WMI battery information call returned buffer of unexpected length %u\n",
+ obj->buffer.length);
+ return -EINVAL;
+ }
+
+ *result = get_unaligned_le32(obj->buffer.pointer);
+
+ return ret;
+}
+
+static int acer_wmi_battery_get_health_control_status(struct acer_wmi_battery_data *data,
+ bool *health_mode)
+{
+ /*
+ * Acer Care Center seems to always call the WMI method
+ * with fixed parameters. This yields information about
+ * the availability and state of both health and
+ * calibration mode. The modes probably apply to
+ * all batteries of the system.
+ */
+ struct get_battery_health_control_status_input args = {
+ .battery_no = ACER_BATTERY_INDEX,
+ .function_query = 0x1,
+ .reserved = { 0x0, 0x0 },
+ };
+ struct acpi_buffer input = { (acpi_size) sizeof(args), &args };
+ struct get_battery_health_control_status_output *status_output;
+ struct acpi_buffer output = { ACPI_ALLOCATE_BUFFER, NULL };
+ int ret;
+
+ ret = wmidev_evaluate_method(data->wdev, 0, 20, &input, &output);
+ if (ACPI_FAILURE(ret))
+ return -EIO;
+
+ union acpi_object *obj __free(kfree) = output.pointer;
+ if (!obj || obj->type != ACPI_TYPE_BUFFER)
+ return -EIO;
+
+ if (obj->buffer.length < sizeof(*status_output)) {
+ dev_err(&data->wdev->dev, "WMI battery status call returned a buffer of unexpected length %d\n",
+ obj->buffer.length);
+ return -EINVAL;
+ }
+
+ status_output = (struct get_battery_health_control_status_output *)obj->buffer.pointer;
+
+ if (health_mode) {
+ if (!(status_output->function_list & HEALTH_MODE))
+ return -EINVAL;
+
+ *health_mode = status_output->function_status[0] > 0;
+ }
+
+ return ret;
+}
+
+static int acer_wmi_battery_set_health_control(struct acer_wmi_battery_data *data,
+ u8 function, bool function_status)
+{
+ struct set_battery_health_control_input args = {
+ .battery_no = ACER_BATTERY_INDEX,
+ .function_mask = function,
+ .function_status = function_status ? 1 : 0,
+ .reserved_in = { 0x0, 0x0, 0x0, 0x0, 0x0 },
+ };
+ struct acpi_buffer input = { (acpi_size) sizeof(args), &args };
+ struct acpi_buffer output = { ACPI_ALLOCATE_BUFFER, NULL };
+ union acpi_object *obj;
+ int ret;
+
+ ret = wmidev_evaluate_method(data->wdev, 0, 21, &input, &output);
+ if (ACPI_FAILURE(ret))
+ return -EIO;
+
+ obj = output.pointer;
+ if (!obj || obj->type != ACPI_TYPE_BUFFER)
+ return -EIO;
+
+ if (obj->buffer.length < 4) {
+ dev_err(&data->wdev->dev, "WMI battery status set operation returned a buffer of unexpected length %d\n",
+ obj->buffer.length);
+ return -EINVAL;
+ }
+
+ return ret;
+}
+
+static int acer_battery_ext_property_get(struct power_supply *psy,
+ const struct power_supply_ext *ext,
+ void *ext_data,
+ enum power_supply_property psp,
+ union power_supply_propval *val)
+{
+ struct acer_wmi_battery_data *data = ext_data;
+ bool health_mode;
+ u32 value;
+ int ret;
+
+ switch (psp) {
+ case POWER_SUPPLY_PROP_CHARGE_TYPES:
+ ret = acer_wmi_battery_get_health_control_status(data, &health_mode);
+ if (ret)
+ return ret;
+
+ val->intval = health_mode ? POWER_SUPPLY_CHARGE_TYPE_LONGLIFE
+ : POWER_SUPPLY_CHARGE_TYPE_STANDARD;
+ break;
+ case POWER_SUPPLY_PROP_TEMP:
+ ret = acer_wmi_battery_get_information(data, 0x8, ACER_BATTERY_INDEX, &value);
+ if (ret)
+ return ret;
+ if (value > U16_MAX)
+ return -ERANGE;
+
+ val->intval = value - 2731;
+ break;
+ default:
+ return -EINVAL;
+ }
+
+ return 0;
+}
+
+static int acer_battery_ext_property_set(struct power_supply *psy,
+ const struct power_supply_ext *ext,
+ void *ext_data,
+ enum power_supply_property psp,
+ const union power_supply_propval *val)
+{
+ struct acer_wmi_battery_data *data = ext_data;
+
+ switch (psp) {
+ case POWER_SUPPLY_PROP_CHARGE_TYPES:
+ return acer_wmi_battery_set_health_control(data, HEALTH_MODE,
+ val->intval == POWER_SUPPLY_CHARGE_TYPE_LONGLIFE);
+ default:
+ return -EINVAL;
+ }
+}
+
+static int acer_battery_ext_property_is_writeable(struct power_supply *psy,
+ const struct power_supply_ext *ext,
+ void *ext_data,
+ enum power_supply_property psp)
+{
+ switch (psp) {
+ case POWER_SUPPLY_PROP_CHARGE_TYPES:
+ return true;
+ default:
+ return false;
+ }
+}
+
+static const struct dmi_system_id acer_wmi_battery_health_mode_table[] = {
+ {
+ .matches = {
+ DMI_MATCH(DMI_SYS_VENDOR, "Acer"),
+ DMI_MATCH(DMI_PRODUCT_NAME, "Aspire A315-510P")
+ }
+ },
+ {}
+};
+
+static const enum power_supply_property acer_battery_properties_v1[] = {
+ POWER_SUPPLY_PROP_TEMP,
+};
+
+static const enum power_supply_property acer_battery_properties_v2[] = {
+ POWER_SUPPLY_PROP_TEMP,
+ POWER_SUPPLY_PROP_CHARGE_TYPES,
+};
+
+static const struct power_supply_ext acer_wmi_battery_extension_v1 = {
+ .name = DRIVER_NAME,
+ .properties = acer_battery_properties_v1,
+ .num_properties = ARRAY_SIZE(acer_battery_properties_v1),
+ .get_property = acer_battery_ext_property_get,
+ .set_property = acer_battery_ext_property_set,
+ .property_is_writeable = acer_battery_ext_property_is_writeable,
+};
+
+static const struct power_supply_ext acer_wmi_battery_extension_v2 = {
+ .name = DRIVER_NAME,
+ .properties = acer_battery_properties_v2,
+ .num_properties = ARRAY_SIZE(acer_battery_properties_v2),
+ .charge_types = BIT(POWER_SUPPLY_CHARGE_TYPE_STANDARD) |
+ BIT(POWER_SUPPLY_CHARGE_TYPE_LONGLIFE),
+ .get_property = acer_battery_ext_property_get,
+ .set_property = acer_battery_ext_property_set,
+ .property_is_writeable = acer_battery_ext_property_is_writeable,
+};
+
+static int acer_battery_add(struct power_supply *battery, struct acpi_battery_hook *hook)
+{
+ struct acer_wmi_battery_data *data = container_of(hook, struct acer_wmi_battery_data, hook);
+
+ return power_supply_register_extension(battery, data->battery_ext,
+ &data->wdev->dev, data);
+}
+
+static int acer_battery_remove(struct power_supply *battery, struct acpi_battery_hook *hook)
+{
+ struct acer_wmi_battery_data *data = container_of(hook, struct acer_wmi_battery_data, hook);
+
+ power_supply_unregister_extension(battery, data->battery_ext);
+
+ return 0;
+}
+
+static int acer_wmi_battery_probe(struct wmi_device *wdev, const void *context)
+{
+ struct acer_wmi_battery_data *data;
+
+ data = devm_kzalloc(&wdev->dev, sizeof(*data), GFP_KERNEL);
+ if (!data)
+ return -ENOMEM;
+
+ dev_set_drvdata(&wdev->dev, data);
+ data->wdev = wdev;
+ data->features.health_mode = dmi_check_system(acer_wmi_battery_health_mode_table);
+ data->battery_ext = data->features.health_mode ? &acer_wmi_battery_extension_v2
+ : &acer_wmi_battery_extension_v1;
+ data->hook.name = "Acer Battery Extension";
+ data->hook.add_battery = acer_battery_add;
+ data->hook.remove_battery = acer_battery_remove;
+
+ return devm_battery_hook_register(&data->wdev->dev, &data->hook);
+}
+
+static const struct wmi_device_id acer_wmi_battery_id_table[] = {
+ { ACER_BATTERY_GUID, NULL },
+ { }
+};
+MODULE_DEVICE_TABLE(wmi, acer_wmi_battery_id_table);
+
+static struct wmi_driver acer_wmi_battery_driver = {
+ .driver = {
+ .name = DRIVER_NAME,
+ .probe_type = PROBE_PREFER_ASYNCHRONOUS,
+ },
+ .id_table = acer_wmi_battery_id_table,
+ .probe = acer_wmi_battery_probe,
+ .no_singleton = true,
+};
+module_wmi_driver(acer_wmi_battery_driver);
+
+MODULE_AUTHOR("Frederik Harwath <frederik@harwath.name>");
+MODULE_AUTHOR("Jelle van der Waa <jelle@vdwaa.nl>");
+MODULE_DESCRIPTION("Acer battery health control WMI driver");
+MODULE_LICENSE("GPL");
--
2.54.0
^ permalink raw reply related [flat|nested] 5+ messages in thread
* Re: [PATCH v3 1/1] platform/x86: add Acer battery control driver
2026-05-10 18:50 ` [PATCH v3 1/1] " Jelle van der Waa
@ 2026-05-11 11:54 ` Ilpo Järvinen
2026-05-11 16:09 ` Rong Zhang
1 sibling, 0 replies; 5+ messages in thread
From: Ilpo Järvinen @ 2026-05-11 11:54 UTC (permalink / raw)
To: Jelle van der Waa
Cc: Hans de Goede, Ilpo Järvinen, platform-driver-x86,
Frederik Harwath
On Sun, 10 May 2026, Jelle van der Waa wrote:
> Some Acer laptops can configure battery related features through Acer
> Care Center on Windows. This driver uses the power supply extension to
> set a battery charge limit and exposes the battery
> temperature.
>
> This driver is based on the existing acer-wmi-battery project on GitHub
> and was tested on an Acer Aspire A315-510P.
>
> Signed-off-by: Jelle van der Waa <jelle@vdwaa.nl>
>
> ---
> v3:
> - Add depends on DMI
> - Rename CamelCase struct member names
> - Simplify returning errors
> - Add comma to non-terminating entries
> - Simplified acer_wmi_battery_set_battery_health_control to acer_wmi_set_battery_health_control
> - Use sizeof for acpi_object buffer
> - Drop POWER_SUPPLY_EXTENSION macro
> v2:
> - Alphabetically sort linux headers
> - Include headers for types / _packed
> - Use cleanup.h instead of goto + label
> - Add missing prefix for set_battery_health_control
> - General code formatting fixes
> - Remove HWMON dependency in Kconfig
> - Use wmidev_evaluate_method()
> - Accept oversized ACPI buffers
> - Use DRIVER_NAME for battery extension name
> - Set no_singleton = true
> - Implement DMI matching to support laptops with only battery
> temperature support.
> ---
> drivers/platform/x86/Kconfig | 12 +
> drivers/platform/x86/Makefile | 1 +
> drivers/platform/x86/acer-wmi-battery.c | 347 ++++++++++++++++++++++++
> 3 files changed, 360 insertions(+)
> create mode 100644 drivers/platform/x86/acer-wmi-battery.c
>
> diff --git a/drivers/platform/x86/Kconfig b/drivers/platform/x86/Kconfig
> index 2ffa4ecf65b0..2fc23cd2039f 100644
> --- a/drivers/platform/x86/Kconfig
> +++ b/drivers/platform/x86/Kconfig
> @@ -188,6 +188,18 @@ config ACER_WMI
> If you have an ACPI-WMI compatible Acer/ Wistron laptop, say Y or M
> here.
>
> +config ACER_WMI_BATTERY
> + tristate "Acer WMI Battery"
> + depends on ACPI_WMI
> + depends on ACPI_BATTERY
> + depends on DMI
> + help
> + This is a driver for Acer laptops with battery health control. It
> + adds charge limit control and battery temperature reporting.
> +
> + If you have an ACPI-WMI Battery compatible Acer laptop, say Y or M
> + here.
> +
> source "drivers/platform/x86/amd/Kconfig"
>
> config ADV_SWBUTTON
> diff --git a/drivers/platform/x86/Makefile b/drivers/platform/x86/Makefile
> index 872ac3842391..a877acd937cd 100644
> --- a/drivers/platform/x86/Makefile
> +++ b/drivers/platform/x86/Makefile
> @@ -20,6 +20,7 @@ obj-$(CONFIG_BITLAND_MIFS_WMI) += bitland-mifs-wmi.o
> obj-$(CONFIG_ACERHDF) += acerhdf.o
> obj-$(CONFIG_ACER_WIRELESS) += acer-wireless.o
> obj-$(CONFIG_ACER_WMI) += acer-wmi.o
> +obj-$(CONFIG_ACER_WMI_BATTERY) += acer-wmi-battery.o
>
> # AMD
> obj-y += amd/
> diff --git a/drivers/platform/x86/acer-wmi-battery.c b/drivers/platform/x86/acer-wmi-battery.c
> new file mode 100644
> index 000000000000..78005e7a7f55
> --- /dev/null
> +++ b/drivers/platform/x86/acer-wmi-battery.c
> @@ -0,0 +1,347 @@
> +// SPDX-License-Identifier: GPL-2.0-or-later
> +/*
> + * acer-wmi-battery.c: Acer battery health control driver
> + *
> + * This is a driver for the WMI battery health control interface found
> + * on some Acer laptops. This interface allows to enable/disable a
> + * battery charge limit ("health mode") and exposes the battery temperature.
> + *
> + * Based on acer-wmi-battery https://github.com/frederik-h/acer-wmi-battery/
> + *
> + * Copyright (C) 2022-2025 Frederik Harwath <frederik@harwath.name>
> + */
> +
> +#include <linux/acpi.h>
> +#include <linux/cleanup.h>
> +#include <linux/compiler_attributes.h>
> +#include <linux/dmi.h>
> +#include <linux/init.h>
> +#include <linux/kernel.h>
> +#include <linux/limits.h>
> +#include <linux/module.h>
> +#include <linux/power_supply.h>
> +#include <linux/types.h>
> +#include <linux/unaligned.h>
> +#include <linux/version.h>
> +#include <linux/wmi.h>
> +
> +#include <acpi/battery.h>
> +
> +#define DRIVER_NAME "acer-wmi-battery"
> +
> +#define ACER_BATTERY_GUID "79772EC5-04B1-4BFD-843C-61E7F77B6CC9"
> +
> +/*
> + * The Acer OEM software seems to always use this battery index,
> + * so we emulate this behaviour to not confuse the underlying firmware.
> + *
> + * However this also means that we only fully support devices with a
> + * single battery for now.
> + */
> +#define ACER_BATTERY_INDEX 0x1
> +
> +struct get_battery_health_control_status_input {
> + u8 battery_no;
> + u8 function_query;
> + u8 reserved[2];
> +} __packed;
> +
> +struct get_battery_health_control_status_output {
> + u8 function_list;
> + u8 ret[2];
> + u8 function_status[5];
> +} __packed;
> +
> +struct set_battery_health_control_input {
> + u8 battery_no;
> + u8 function_mask;
> + u8 function_status;
> + u8 reserved_in[5];
> +} __packed;
> +
> +struct set_battery_health_control_output {
> + u8 ret;
> + u8 reserved_out;
> +} __packed;
> +
> +enum battery_mode {
> + HEALTH_MODE = 1,
> + CALIBRATION_MODE = 2,
> +};
> +
> +struct acer_wmi_battery_data {
> + struct acpi_battery_hook hook;
> + struct wmi_device *wdev;
> + const struct power_supply_ext *battery_ext;
> + struct {
> + bool health_mode : 1;
> + } features;
> +};
> +
> +static int acer_wmi_battery_get_information(struct acer_wmi_battery_data *data,
> + u32 index, u32 battery, u32 *result)
> +{
> + u32 args[2] = { index, battery };
> + struct acpi_buffer output = { ACPI_ALLOCATE_BUFFER, NULL };
> + struct acpi_buffer input = { sizeof(args), args };
> + int ret;
> +
> + ret = wmidev_evaluate_method(data->wdev, 0, 19, &input, &output);
Hi,
Armin has made improvements to WMI API, and as a result, this interface
has been deprecated:
/**
* wmidev_evaluate_method - Evaluate a WMI method (deprecated)
(FYI, there's one fix to the new WMI interface currently only in fixes
branch which I plan to merge to for-next once the next fixes PR is done,
in case you end up hitting that issue in the meantime.)
--
i.
> + if (ACPI_FAILURE(ret))
> + return -EIO;
> +
> + union acpi_object *obj __free(kfree) = output.pointer;
> + if (!obj || obj->type != ACPI_TYPE_BUFFER)
> + return -EIO;
> +
> + if (obj->buffer.length < sizeof(u32)) {
> + dev_err(&data->wdev->dev, "WMI battery information call returned buffer of unexpected length %u\n",
> + obj->buffer.length);
> + return -EINVAL;
> + }
> +
> + *result = get_unaligned_le32(obj->buffer.pointer);
> +
> + return ret;
> +}
> +
> +static int acer_wmi_battery_get_health_control_status(struct acer_wmi_battery_data *data,
> + bool *health_mode)
> +{
> + /*
> + * Acer Care Center seems to always call the WMI method
> + * with fixed parameters. This yields information about
> + * the availability and state of both health and
> + * calibration mode. The modes probably apply to
> + * all batteries of the system.
> + */
> + struct get_battery_health_control_status_input args = {
> + .battery_no = ACER_BATTERY_INDEX,
> + .function_query = 0x1,
> + .reserved = { 0x0, 0x0 },
> + };
> + struct acpi_buffer input = { (acpi_size) sizeof(args), &args };
> + struct get_battery_health_control_status_output *status_output;
> + struct acpi_buffer output = { ACPI_ALLOCATE_BUFFER, NULL };
> + int ret;
> +
> + ret = wmidev_evaluate_method(data->wdev, 0, 20, &input, &output);
> + if (ACPI_FAILURE(ret))
> + return -EIO;
> +
> + union acpi_object *obj __free(kfree) = output.pointer;
> + if (!obj || obj->type != ACPI_TYPE_BUFFER)
> + return -EIO;
> +
> + if (obj->buffer.length < sizeof(*status_output)) {
> + dev_err(&data->wdev->dev, "WMI battery status call returned a buffer of unexpected length %d\n",
> + obj->buffer.length);
> + return -EINVAL;
> + }
> +
> + status_output = (struct get_battery_health_control_status_output *)obj->buffer.pointer;
> +
> + if (health_mode) {
> + if (!(status_output->function_list & HEALTH_MODE))
> + return -EINVAL;
> +
> + *health_mode = status_output->function_status[0] > 0;
> + }
> +
> + return ret;
> +}
> +
> +static int acer_wmi_battery_set_health_control(struct acer_wmi_battery_data *data,
> + u8 function, bool function_status)
> +{
> + struct set_battery_health_control_input args = {
> + .battery_no = ACER_BATTERY_INDEX,
> + .function_mask = function,
> + .function_status = function_status ? 1 : 0,
> + .reserved_in = { 0x0, 0x0, 0x0, 0x0, 0x0 },
> + };
> + struct acpi_buffer input = { (acpi_size) sizeof(args), &args };
> + struct acpi_buffer output = { ACPI_ALLOCATE_BUFFER, NULL };
> + union acpi_object *obj;
> + int ret;
> +
> + ret = wmidev_evaluate_method(data->wdev, 0, 21, &input, &output);
> + if (ACPI_FAILURE(ret))
> + return -EIO;
> +
> + obj = output.pointer;
> + if (!obj || obj->type != ACPI_TYPE_BUFFER)
> + return -EIO;
> +
> + if (obj->buffer.length < 4) {
> + dev_err(&data->wdev->dev, "WMI battery status set operation returned a buffer of unexpected length %d\n",
> + obj->buffer.length);
> + return -EINVAL;
> + }
> +
> + return ret;
> +}
> +
> +static int acer_battery_ext_property_get(struct power_supply *psy,
> + const struct power_supply_ext *ext,
> + void *ext_data,
> + enum power_supply_property psp,
> + union power_supply_propval *val)
> +{
> + struct acer_wmi_battery_data *data = ext_data;
> + bool health_mode;
> + u32 value;
> + int ret;
> +
> + switch (psp) {
> + case POWER_SUPPLY_PROP_CHARGE_TYPES:
> + ret = acer_wmi_battery_get_health_control_status(data, &health_mode);
> + if (ret)
> + return ret;
> +
> + val->intval = health_mode ? POWER_SUPPLY_CHARGE_TYPE_LONGLIFE
> + : POWER_SUPPLY_CHARGE_TYPE_STANDARD;
> + break;
> + case POWER_SUPPLY_PROP_TEMP:
> + ret = acer_wmi_battery_get_information(data, 0x8, ACER_BATTERY_INDEX, &value);
> + if (ret)
> + return ret;
> + if (value > U16_MAX)
> + return -ERANGE;
> +
> + val->intval = value - 2731;
> + break;
> + default:
> + return -EINVAL;
> + }
> +
> + return 0;
> +}
> +
> +static int acer_battery_ext_property_set(struct power_supply *psy,
> + const struct power_supply_ext *ext,
> + void *ext_data,
> + enum power_supply_property psp,
> + const union power_supply_propval *val)
> +{
> + struct acer_wmi_battery_data *data = ext_data;
> +
> + switch (psp) {
> + case POWER_SUPPLY_PROP_CHARGE_TYPES:
> + return acer_wmi_battery_set_health_control(data, HEALTH_MODE,
> + val->intval == POWER_SUPPLY_CHARGE_TYPE_LONGLIFE);
> + default:
> + return -EINVAL;
> + }
> +}
> +
> +static int acer_battery_ext_property_is_writeable(struct power_supply *psy,
> + const struct power_supply_ext *ext,
> + void *ext_data,
> + enum power_supply_property psp)
> +{
> + switch (psp) {
> + case POWER_SUPPLY_PROP_CHARGE_TYPES:
> + return true;
> + default:
> + return false;
> + }
> +}
> +
> +static const struct dmi_system_id acer_wmi_battery_health_mode_table[] = {
> + {
> + .matches = {
> + DMI_MATCH(DMI_SYS_VENDOR, "Acer"),
> + DMI_MATCH(DMI_PRODUCT_NAME, "Aspire A315-510P")
> + }
> + },
> + {}
> +};
> +
> +static const enum power_supply_property acer_battery_properties_v1[] = {
> + POWER_SUPPLY_PROP_TEMP,
> +};
> +
> +static const enum power_supply_property acer_battery_properties_v2[] = {
> + POWER_SUPPLY_PROP_TEMP,
> + POWER_SUPPLY_PROP_CHARGE_TYPES,
> +};
> +
> +static const struct power_supply_ext acer_wmi_battery_extension_v1 = {
> + .name = DRIVER_NAME,
> + .properties = acer_battery_properties_v1,
> + .num_properties = ARRAY_SIZE(acer_battery_properties_v1),
> + .get_property = acer_battery_ext_property_get,
> + .set_property = acer_battery_ext_property_set,
> + .property_is_writeable = acer_battery_ext_property_is_writeable,
> +};
> +
> +static const struct power_supply_ext acer_wmi_battery_extension_v2 = {
> + .name = DRIVER_NAME,
> + .properties = acer_battery_properties_v2,
> + .num_properties = ARRAY_SIZE(acer_battery_properties_v2),
> + .charge_types = BIT(POWER_SUPPLY_CHARGE_TYPE_STANDARD) |
> + BIT(POWER_SUPPLY_CHARGE_TYPE_LONGLIFE),
> + .get_property = acer_battery_ext_property_get,
> + .set_property = acer_battery_ext_property_set,
> + .property_is_writeable = acer_battery_ext_property_is_writeable,
> +};
> +
> +static int acer_battery_add(struct power_supply *battery, struct acpi_battery_hook *hook)
> +{
> + struct acer_wmi_battery_data *data = container_of(hook, struct acer_wmi_battery_data, hook);
> +
> + return power_supply_register_extension(battery, data->battery_ext,
> + &data->wdev->dev, data);
> +}
> +
> +static int acer_battery_remove(struct power_supply *battery, struct acpi_battery_hook *hook)
> +{
> + struct acer_wmi_battery_data *data = container_of(hook, struct acer_wmi_battery_data, hook);
> +
> + power_supply_unregister_extension(battery, data->battery_ext);
> +
> + return 0;
> +}
> +
> +static int acer_wmi_battery_probe(struct wmi_device *wdev, const void *context)
> +{
> + struct acer_wmi_battery_data *data;
> +
> + data = devm_kzalloc(&wdev->dev, sizeof(*data), GFP_KERNEL);
> + if (!data)
> + return -ENOMEM;
> +
> + dev_set_drvdata(&wdev->dev, data);
> + data->wdev = wdev;
> + data->features.health_mode = dmi_check_system(acer_wmi_battery_health_mode_table);
> + data->battery_ext = data->features.health_mode ? &acer_wmi_battery_extension_v2
> + : &acer_wmi_battery_extension_v1;
> + data->hook.name = "Acer Battery Extension";
> + data->hook.add_battery = acer_battery_add;
> + data->hook.remove_battery = acer_battery_remove;
> +
> + return devm_battery_hook_register(&data->wdev->dev, &data->hook);
> +}
> +
> +static const struct wmi_device_id acer_wmi_battery_id_table[] = {
> + { ACER_BATTERY_GUID, NULL },
> + { }
> +};
> +MODULE_DEVICE_TABLE(wmi, acer_wmi_battery_id_table);
> +
> +static struct wmi_driver acer_wmi_battery_driver = {
> + .driver = {
> + .name = DRIVER_NAME,
> + .probe_type = PROBE_PREFER_ASYNCHRONOUS,
> + },
> + .id_table = acer_wmi_battery_id_table,
> + .probe = acer_wmi_battery_probe,
> + .no_singleton = true,
> +};
> +module_wmi_driver(acer_wmi_battery_driver);
> +
> +MODULE_AUTHOR("Frederik Harwath <frederik@harwath.name>");
> +MODULE_AUTHOR("Jelle van der Waa <jelle@vdwaa.nl>");
> +MODULE_DESCRIPTION("Acer battery health control WMI driver");
> +MODULE_LICENSE("GPL");
>
^ permalink raw reply [flat|nested] 5+ messages in thread
* Re: [PATCH v3 1/1] platform/x86: add Acer battery control driver
2026-05-10 18:50 ` [PATCH v3 1/1] " Jelle van der Waa
2026-05-11 11:54 ` Ilpo Järvinen
@ 2026-05-11 16:09 ` Rong Zhang
2026-05-11 18:53 ` Hans de Goede
1 sibling, 1 reply; 5+ messages in thread
From: Rong Zhang @ 2026-05-11 16:09 UTC (permalink / raw)
To: Jelle van der Waa, Hans de Goede, Ilpo Järvinen
Cc: platform-driver-x86, Frederik Harwath
Hi Jelle,
On Sun, 2026-05-10 at 20:50 +0200, Jelle van der Waa wrote:
> Some Acer laptops can configure battery related features through Acer
> Care Center on Windows. This driver uses the power supply extension to
> set a battery charge limit
>
What does it limit? If it limits the charge capacity, ...
> and exposes the battery
> temperature.
>
> This driver is based on the existing acer-wmi-battery project on GitHub
> and was tested on an Acer Aspire A315-510P.
>
> Signed-off-by: Jelle van der Waa <jelle@vdwaa.nl>
>
> ---
> v3:
> - Add depends on DMI
> - Rename CamelCase struct member names
> - Simplify returning errors
> - Add comma to non-terminating entries
> - Simplified acer_wmi_battery_set_battery_health_control to acer_wmi_set_battery_health_control
> - Use sizeof for acpi_object buffer
> - Drop POWER_SUPPLY_EXTENSION macro
> v2:
> - Alphabetically sort linux headers
> - Include headers for types / _packed
> - Use cleanup.h instead of goto + label
> - Add missing prefix for set_battery_health_control
> - General code formatting fixes
> - Remove HWMON dependency in Kconfig
> - Use wmidev_evaluate_method()
> - Accept oversized ACPI buffers
> - Use DRIVER_NAME for battery extension name
> - Set no_singleton = true
> - Implement DMI matching to support laptops with only battery
> temperature support.
> ---
> drivers/platform/x86/Kconfig | 12 +
> drivers/platform/x86/Makefile | 1 +
> drivers/platform/x86/acer-wmi-battery.c | 347 ++++++++++++++++++++++++
> 3 files changed, 360 insertions(+)
> create mode 100644 drivers/platform/x86/acer-wmi-battery.c
>
> diff --git a/drivers/platform/x86/Kconfig b/drivers/platform/x86/Kconfig
> index 2ffa4ecf65b0..2fc23cd2039f 100644
> --- a/drivers/platform/x86/Kconfig
> +++ b/drivers/platform/x86/Kconfig
> @@ -188,6 +188,18 @@ config ACER_WMI
> If you have an ACPI-WMI compatible Acer/ Wistron laptop, say Y or M
> here.
>
> +config ACER_WMI_BATTERY
> + tristate "Acer WMI Battery"
> + depends on ACPI_WMI
> + depends on ACPI_BATTERY
> + depends on DMI
> + help
> + This is a driver for Acer laptops with battery health control. It
> + adds charge limit control and battery temperature reporting.
> +
> + If you have an ACPI-WMI Battery compatible Acer laptop, say Y or M
> + here.
> +
> source "drivers/platform/x86/amd/Kconfig"
>
> config ADV_SWBUTTON
> diff --git a/drivers/platform/x86/Makefile b/drivers/platform/x86/Makefile
> index 872ac3842391..a877acd937cd 100644
> --- a/drivers/platform/x86/Makefile
> +++ b/drivers/platform/x86/Makefile
> @@ -20,6 +20,7 @@ obj-$(CONFIG_BITLAND_MIFS_WMI) += bitland-mifs-wmi.o
> obj-$(CONFIG_ACERHDF) += acerhdf.o
> obj-$(CONFIG_ACER_WIRELESS) += acer-wireless.o
> obj-$(CONFIG_ACER_WMI) += acer-wmi.o
> +obj-$(CONFIG_ACER_WMI_BATTERY) += acer-wmi-battery.o
>
> # AMD
> obj-y += amd/
> diff --git a/drivers/platform/x86/acer-wmi-battery.c b/drivers/platform/x86/acer-wmi-battery.c
> new file mode 100644
> index 000000000000..78005e7a7f55
> --- /dev/null
> +++ b/drivers/platform/x86/acer-wmi-battery.c
> @@ -0,0 +1,347 @@
> +// SPDX-License-Identifier: GPL-2.0-or-later
> +/*
> + * acer-wmi-battery.c: Acer battery health control driver
> + *
> + * This is a driver for the WMI battery health control interface found
> + * on some Acer laptops. This interface allows to enable/disable a
> + * battery charge limit ("health mode") and exposes the battery temperature.
> + *
> + * Based on acer-wmi-battery https://github.com/frederik-h/acer-wmi-battery/
> + *
> + * Copyright (C) 2022-2025 Frederik Harwath <frederik@harwath.name>
> + */
> +
> +#include <linux/acpi.h>
> +#include <linux/cleanup.h>
> +#include <linux/compiler_attributes.h>
> +#include <linux/dmi.h>
> +#include <linux/init.h>
> +#include <linux/kernel.h>
> +#include <linux/limits.h>
> +#include <linux/module.h>
> +#include <linux/power_supply.h>
> +#include <linux/types.h>
> +#include <linux/unaligned.h>
> +#include <linux/version.h>
> +#include <linux/wmi.h>
> +
> +#include <acpi/battery.h>
> +
> +#define DRIVER_NAME "acer-wmi-battery"
> +
> +#define ACER_BATTERY_GUID "79772EC5-04B1-4BFD-843C-61E7F77B6CC9"
> +
> +/*
> + * The Acer OEM software seems to always use this battery index,
> + * so we emulate this behaviour to not confuse the underlying firmware.
> + *
> + * However this also means that we only fully support devices with a
> + * single battery for now.
> + */
> +#define ACER_BATTERY_INDEX 0x1
> +
> +struct get_battery_health_control_status_input {
> + u8 battery_no;
> + u8 function_query;
> + u8 reserved[2];
> +} __packed;
> +
> +struct get_battery_health_control_status_output {
> + u8 function_list;
> + u8 ret[2];
> + u8 function_status[5];
> +} __packed;
> +
> +struct set_battery_health_control_input {
> + u8 battery_no;
> + u8 function_mask;
> + u8 function_status;
> + u8 reserved_in[5];
> +} __packed;
> +
> +struct set_battery_health_control_output {
> + u8 ret;
> + u8 reserved_out;
> +} __packed;
> +
> +enum battery_mode {
> + HEALTH_MODE = 1,
> + CALIBRATION_MODE = 2,
> +};
> +
> +struct acer_wmi_battery_data {
> + struct acpi_battery_hook hook;
> + struct wmi_device *wdev;
> + const struct power_supply_ext *battery_ext;
> + struct {
> + bool health_mode : 1;
> + } features;
> +};
> +
> +static int acer_wmi_battery_get_information(struct acer_wmi_battery_data *data,
> + u32 index, u32 battery, u32 *result)
> +{
> + u32 args[2] = { index, battery };
> + struct acpi_buffer output = { ACPI_ALLOCATE_BUFFER, NULL };
> + struct acpi_buffer input = { sizeof(args), args };
> + int ret;
> +
> + ret = wmidev_evaluate_method(data->wdev, 0, 19, &input, &output);
> + if (ACPI_FAILURE(ret))
> + return -EIO;
> +
> + union acpi_object *obj __free(kfree) = output.pointer;
> + if (!obj || obj->type != ACPI_TYPE_BUFFER)
> + return -EIO;
> +
> + if (obj->buffer.length < sizeof(u32)) {
> + dev_err(&data->wdev->dev, "WMI battery information call returned buffer of unexpected length %u\n",
> + obj->buffer.length);
> + return -EINVAL;
> + }
> +
> + *result = get_unaligned_le32(obj->buffer.pointer);
> +
> + return ret;
> +}
> +
> +static int acer_wmi_battery_get_health_control_status(struct acer_wmi_battery_data *data,
> + bool *health_mode)
> +{
> + /*
> + * Acer Care Center seems to always call the WMI method
> + * with fixed parameters. This yields information about
> + * the availability and state of both health and
> + * calibration mode. The modes probably apply to
> + * all batteries of the system.
> + */
> + struct get_battery_health_control_status_input args = {
> + .battery_no = ACER_BATTERY_INDEX,
> + .function_query = 0x1,
> + .reserved = { 0x0, 0x0 },
> + };
> + struct acpi_buffer input = { (acpi_size) sizeof(args), &args };
> + struct get_battery_health_control_status_output *status_output;
> + struct acpi_buffer output = { ACPI_ALLOCATE_BUFFER, NULL };
> + int ret;
> +
> + ret = wmidev_evaluate_method(data->wdev, 0, 20, &input, &output);
> + if (ACPI_FAILURE(ret))
> + return -EIO;
> +
> + union acpi_object *obj __free(kfree) = output.pointer;
> + if (!obj || obj->type != ACPI_TYPE_BUFFER)
> + return -EIO;
> +
> + if (obj->buffer.length < sizeof(*status_output)) {
> + dev_err(&data->wdev->dev, "WMI battery status call returned a buffer of unexpected length %d\n",
> + obj->buffer.length);
> + return -EINVAL;
> + }
> +
> + status_output = (struct get_battery_health_control_status_output *)obj->buffer.pointer;
> +
> + if (health_mode) {
> + if (!(status_output->function_list & HEALTH_MODE))
> + return -EINVAL;
> +
> + *health_mode = status_output->function_status[0] > 0;
> + }
> +
> + return ret;
> +}
> +
> +static int acer_wmi_battery_set_health_control(struct acer_wmi_battery_data *data,
> + u8 function, bool function_status)
> +{
> + struct set_battery_health_control_input args = {
> + .battery_no = ACER_BATTERY_INDEX,
> + .function_mask = function,
> + .function_status = function_status ? 1 : 0,
> + .reserved_in = { 0x0, 0x0, 0x0, 0x0, 0x0 },
> + };
> + struct acpi_buffer input = { (acpi_size) sizeof(args), &args };
> + struct acpi_buffer output = { ACPI_ALLOCATE_BUFFER, NULL };
> + union acpi_object *obj;
> + int ret;
> +
> + ret = wmidev_evaluate_method(data->wdev, 0, 21, &input, &output);
> + if (ACPI_FAILURE(ret))
> + return -EIO;
> +
> + obj = output.pointer;
> + if (!obj || obj->type != ACPI_TYPE_BUFFER)
> + return -EIO;
> +
> + if (obj->buffer.length < 4) {
> + dev_err(&data->wdev->dev, "WMI battery status set operation returned a buffer of unexpected length %d\n",
> + obj->buffer.length);
> + return -EINVAL;
> + }
> +
> + return ret;
> +}
> +
> +static int acer_battery_ext_property_get(struct power_supply *psy,
> + const struct power_supply_ext *ext,
> + void *ext_data,
> + enum power_supply_property psp,
> + union power_supply_propval *val)
> +{
> + struct acer_wmi_battery_data *data = ext_data;
> + bool health_mode;
> + u32 value;
> + int ret;
> +
> + switch (psp) {
> + case POWER_SUPPLY_PROP_CHARGE_TYPES:
> + ret = acer_wmi_battery_get_health_control_status(data, &health_mode);
> + if (ret)
> + return ret;
> +
> + val->intval = health_mode ? POWER_SUPPLY_CHARGE_TYPE_LONGLIFE
> + : POWER_SUPPLY_CHARGE_TYPE_STANDARD;
> + break;
> + case POWER_SUPPLY_PROP_TEMP:
> + ret = acer_wmi_battery_get_information(data, 0x8, ACER_BATTERY_INDEX, &value);
> + if (ret)
> + return ret;
> + if (value > U16_MAX)
> + return -ERANGE;
> +
> + val->intval = value - 2731;
> + break;
> + default:
> + return -EINVAL;
> + }
> +
> + return 0;
> +}
> +
> +static int acer_battery_ext_property_set(struct power_supply *psy,
> + const struct power_supply_ext *ext,
> + void *ext_data,
> + enum power_supply_property psp,
> + const union power_supply_propval *val)
> +{
> + struct acer_wmi_battery_data *data = ext_data;
> +
> + switch (psp) {
> + case POWER_SUPPLY_PROP_CHARGE_TYPES:
> + return acer_wmi_battery_set_health_control(data, HEALTH_MODE,
> + val->intval == POWER_SUPPLY_CHARGE_TYPE_LONGLIFE);
> + default:
> + return -EINVAL;
> + }
> +}
> +
> +static int acer_battery_ext_property_is_writeable(struct power_supply *psy,
> + const struct power_supply_ext *ext,
> + void *ext_data,
> + enum power_supply_property psp)
> +{
> + switch (psp) {
> + case POWER_SUPPLY_PROP_CHARGE_TYPES:
> + return true;
> + default:
> + return false;
> + }
> +}
> +
> +static const struct dmi_system_id acer_wmi_battery_health_mode_table[] = {
> + {
> + .matches = {
> + DMI_MATCH(DMI_SYS_VENDOR, "Acer"),
> + DMI_MATCH(DMI_PRODUCT_NAME, "Aspire A315-510P")
> + }
> + },
> + {}
> +};
> +
> +static const enum power_supply_property acer_battery_properties_v1[] = {
> + POWER_SUPPLY_PROP_TEMP,
> +};
> +
> +static const enum power_supply_property acer_battery_properties_v2[] = {
> + POWER_SUPPLY_PROP_TEMP,
> + POWER_SUPPLY_PROP_CHARGE_TYPES,
> +};
> +
> +static const struct power_supply_ext acer_wmi_battery_extension_v1 = {
> + .name = DRIVER_NAME,
> + .properties = acer_battery_properties_v1,
> + .num_properties = ARRAY_SIZE(acer_battery_properties_v1),
> + .get_property = acer_battery_ext_property_get,
> + .set_property = acer_battery_ext_property_set,
> + .property_is_writeable = acer_battery_ext_property_is_writeable,
> +};
> +
> +static const struct power_supply_ext acer_wmi_battery_extension_v2 = {
> + .name = DRIVER_NAME,
> + .properties = acer_battery_properties_v2,
> + .num_properties = ARRAY_SIZE(acer_battery_properties_v2),
> + .charge_types = BIT(POWER_SUPPLY_CHARGE_TYPE_STANDARD) |
> + BIT(POWER_SUPPLY_CHARGE_TYPE_LONGLIFE),
I recently learned that POWER_SUPPLY_CHARGE_TYPE_LONGLIFE is not the
correct uapi for charge capacity limit. See
https://lore.kernel.org/r/09d21c9cee8af49fa1d5b568358db0c347668ee9.camel@collabora.com/
Thanks,
Rong
> + .get_property = acer_battery_ext_property_get,
> + .set_property = acer_battery_ext_property_set,
> + .property_is_writeable = acer_battery_ext_property_is_writeable,
> +};
> +
> +static int acer_battery_add(struct power_supply *battery, struct acpi_battery_hook *hook)
> +{
> + struct acer_wmi_battery_data *data = container_of(hook, struct acer_wmi_battery_data, hook);
> +
> + return power_supply_register_extension(battery, data->battery_ext,
> + &data->wdev->dev, data);
> +}
> +
> +static int acer_battery_remove(struct power_supply *battery, struct acpi_battery_hook *hook)
> +{
> + struct acer_wmi_battery_data *data = container_of(hook, struct acer_wmi_battery_data, hook);
> +
> + power_supply_unregister_extension(battery, data->battery_ext);
> +
> + return 0;
> +}
> +
> +static int acer_wmi_battery_probe(struct wmi_device *wdev, const void *context)
> +{
> + struct acer_wmi_battery_data *data;
> +
> + data = devm_kzalloc(&wdev->dev, sizeof(*data), GFP_KERNEL);
> + if (!data)
> + return -ENOMEM;
> +
> + dev_set_drvdata(&wdev->dev, data);
> + data->wdev = wdev;
> + data->features.health_mode = dmi_check_system(acer_wmi_battery_health_mode_table);
> + data->battery_ext = data->features.health_mode ? &acer_wmi_battery_extension_v2
> + : &acer_wmi_battery_extension_v1;
> + data->hook.name = "Acer Battery Extension";
> + data->hook.add_battery = acer_battery_add;
> + data->hook.remove_battery = acer_battery_remove;
> +
> + return devm_battery_hook_register(&data->wdev->dev, &data->hook);
> +}
> +
> +static const struct wmi_device_id acer_wmi_battery_id_table[] = {
> + { ACER_BATTERY_GUID, NULL },
> + { }
> +};
> +MODULE_DEVICE_TABLE(wmi, acer_wmi_battery_id_table);
> +
> +static struct wmi_driver acer_wmi_battery_driver = {
> + .driver = {
> + .name = DRIVER_NAME,
> + .probe_type = PROBE_PREFER_ASYNCHRONOUS,
> + },
> + .id_table = acer_wmi_battery_id_table,
> + .probe = acer_wmi_battery_probe,
> + .no_singleton = true,
> +};
> +module_wmi_driver(acer_wmi_battery_driver);
> +
> +MODULE_AUTHOR("Frederik Harwath <frederik@harwath.name>");
> +MODULE_AUTHOR("Jelle van der Waa <jelle@vdwaa.nl>");
> +MODULE_DESCRIPTION("Acer battery health control WMI driver");
> +MODULE_LICENSE("GPL");
^ permalink raw reply [flat|nested] 5+ messages in thread
* Re: [PATCH v3 1/1] platform/x86: add Acer battery control driver
2026-05-11 16:09 ` Rong Zhang
@ 2026-05-11 18:53 ` Hans de Goede
0 siblings, 0 replies; 5+ messages in thread
From: Hans de Goede @ 2026-05-11 18:53 UTC (permalink / raw)
To: Rong Zhang, Jelle van der Waa, Ilpo Järvinen
Cc: platform-driver-x86, Frederik Harwath
Hi Rong,
On 11-May-26 18:09, Rong Zhang wrote:
> Hi Jelle,
>
> On Sun, 2026-05-10 at 20:50 +0200, Jelle van der Waa wrote:
>> Some Acer laptops can configure battery related features through Acer
>> Care Center on Windows. This driver uses the power supply extension to
>> set a battery charge limit
>>
>
> What does it limit? If it limits the charge capacity, ...
>
>> and exposes the battery
<snip>
> I recently learned that POWER_SUPPLY_CHARGE_TYPE_LONGLIFE is not the
> correct uapi for charge capacity limit. See
> https://lore.kernel.org/r/09d21c9cee8af49fa1d5b568358db0c347668ee9.camel@collabora.com/
As first dicussed here and acked by Sebastian Reichel the
power-supply subsys maintainer using charge_types exposing
standard + long-life is the correct thing to do when there
is a boolean long-life on/off toggle rather then a configurable
stop-charging percentage setting, see:
https://lore.kernel.org/linux-pm/49993a42-aa91-46bf-acef-4a089db4c2db@redhat.com/
https://lore.kernel.org/platform-driver-x86/20241209204051.8786-1-hdegoede@redhat.com/
and also here, which dicusses how upower now also supports
using charge_types for this.
https://vdwaa.nl/charge-types-api-upower.html
I've also replied in the thread.
TL;DR: your original approach to use charge_types in your
patch is correct.
Regards,
Hans
^ permalink raw reply [flat|nested] 5+ messages in thread
end of thread, other threads:[~2026-05-11 18:53 UTC | newest]
Thread overview: 5+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2026-05-10 18:50 [PATCH v3 0/1] platform/x86: add Acer battery control driver Jelle van der Waa
2026-05-10 18:50 ` [PATCH v3 1/1] " Jelle van der Waa
2026-05-11 11:54 ` Ilpo Järvinen
2026-05-11 16:09 ` Rong Zhang
2026-05-11 18:53 ` Hans de Goede
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.