[PATCH 4/9] platform/x86: dell-wmi-base: Use new buffer-based WMI API

From: Armin Wolf

Date: Sat Mar 07 2026 - 19:26:01 EST


Use the new buffer-based WMI API to also support ACPI firmware
implementations that do not use ACPI buffers for the event data.

Signed-off-by: Armin Wolf <W_Armin@xxxxxx>
---
drivers/platform/x86/dell/dell-wmi-base.c | 68 ++++++++++++-----------
1 file changed, 35 insertions(+), 33 deletions(-)

diff --git a/drivers/platform/x86/dell/dell-wmi-base.c b/drivers/platform/x86/dell/dell-wmi-base.c
index 4eefbade2f5e..4a7ab9fb3f81 100644
--- a/drivers/platform/x86/dell/dell-wmi-base.c
+++ b/drivers/platform/x86/dell/dell-wmi-base.c
@@ -13,6 +13,7 @@

#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt

+#include <linux/compiler_attributes.h>
#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/init.h>
@@ -408,7 +409,8 @@ static void dell_wmi_switch_event(struct input_dev **subdev,
input_sync(*subdev);
}

-static int dell_wmi_process_key(struct wmi_device *wdev, int type, int code, u16 *buffer, int remaining)
+static int dell_wmi_process_key(struct wmi_device *wdev, int type, int code, __le16 *buffer,
+ int remaining)
{
struct dell_wmi_priv *priv = dev_get_drvdata(&wdev->dev);
const struct key_entry *key;
@@ -440,15 +442,15 @@ static int dell_wmi_process_key(struct wmi_device *wdev, int type, int code, u16
} else if (type == 0x0011 && code == 0xe070 && remaining > 0) {
dell_wmi_switch_event(&priv->tabletswitch_dev,
"Dell tablet mode switch",
- SW_TABLET_MODE, !buffer[0]);
+ SW_TABLET_MODE, !le16_to_cpu(buffer[0]));
return 1;
} else if (type == 0x0012 && code == 0x000c && remaining > 0) {
/* Eprivacy toggle, switch to "on" key entry for on events */
- if (buffer[0] == 2)
+ if (le16_to_cpu(buffer[0]) == 2)
key++;
used = 1;
} else if (type == 0x0012 && code == 0x000d && remaining > 0) {
- value = (buffer[2] == 2);
+ value = (le16_to_cpu(buffer[2]) == 2);
used = 1;
}

@@ -457,24 +459,17 @@ static int dell_wmi_process_key(struct wmi_device *wdev, int type, int code, u16
return used;
}

-static void dell_wmi_notify(struct wmi_device *wdev,
- union acpi_object *obj)
+static void dell_wmi_notify(struct wmi_device *wdev, const struct wmi_buffer *buffer)
{
struct dell_wmi_priv *priv = dev_get_drvdata(&wdev->dev);
- u16 *buffer_entry, *buffer_end;
- acpi_size buffer_size;
+ __le16 *buffer_entry, *buffer_end;
+ size_t buffer_size;
int len, i;

- if (obj->type != ACPI_TYPE_BUFFER) {
- pr_warn("bad response type %x\n", obj->type);
- return;
- }
+ pr_debug("Received WMI event (%*ph)\n", (int)buffer->length, buffer->data);

- pr_debug("Received WMI event (%*ph)\n",
- obj->buffer.length, obj->buffer.pointer);
-
- buffer_entry = (u16 *)obj->buffer.pointer;
- buffer_size = obj->buffer.length/2;
+ buffer_entry = buffer->data;
+ buffer_size = buffer->length / 2;
buffer_end = buffer_entry + buffer_size;

/*
@@ -490,12 +485,12 @@ static void dell_wmi_notify(struct wmi_device *wdev,
* one event on devices with WMI interface version 0.
*/
if (priv->interface_version == 0 && buffer_entry < buffer_end)
- if (buffer_end > buffer_entry + buffer_entry[0] + 1)
- buffer_end = buffer_entry + buffer_entry[0] + 1;
+ if (buffer_end > buffer_entry + le16_to_cpu(buffer_entry[0]) + 1)
+ buffer_end = buffer_entry + le16_to_cpu(buffer_entry[0]) + 1;

while (buffer_entry < buffer_end) {

- len = buffer_entry[0];
+ len = le16_to_cpu(buffer_entry[0]);
if (len == 0)
break;

@@ -508,11 +503,11 @@ static void dell_wmi_notify(struct wmi_device *wdev,

pr_debug("Process buffer (%*ph)\n", len*2, buffer_entry);

- switch (buffer_entry[1]) {
+ switch (le16_to_cpu(buffer_entry[1])) {
case 0x0000: /* One key pressed or event occurred */
if (len > 2)
- dell_wmi_process_key(wdev, buffer_entry[1],
- buffer_entry[2],
+ dell_wmi_process_key(wdev, le16_to_cpu(buffer_entry[1]),
+ le16_to_cpu(buffer_entry[2]),
buffer_entry + 3,
len - 3);
/* Extended data is currently ignored */
@@ -520,22 +515,29 @@ static void dell_wmi_notify(struct wmi_device *wdev,
case 0x0010: /* Sequence of keys pressed */
case 0x0011: /* Sequence of events occurred */
for (i = 2; i < len; ++i)
- i += dell_wmi_process_key(wdev, buffer_entry[1],
- buffer_entry[i],
+ i += dell_wmi_process_key(wdev, le16_to_cpu(buffer_entry[1]),
+ le16_to_cpu(buffer_entry[i]),
buffer_entry + i,
len - i - 1);
break;
case 0x0012:
- if ((len > 4) && dell_privacy_process_event(buffer_entry[1], buffer_entry[3],
- buffer_entry[4]))
- /* dell_privacy_process_event has handled the event */;
- else if (len > 2)
- dell_wmi_process_key(wdev, buffer_entry[1], buffer_entry[2],
+ if (len > 4) {
+ if (dell_privacy_process_event(le16_to_cpu(buffer_entry[1]),
+ le16_to_cpu(buffer_entry[3]),
+ le16_to_cpu(buffer_entry[4])))
+ break;
+ }
+
+ /* dell_privacy_process_event has not handled the event */
+
+ if (len > 2)
+ dell_wmi_process_key(wdev, le16_to_cpu(buffer_entry[1]),
+ le16_to_cpu(buffer_entry[2]),
buffer_entry + 3, len - 3);
+
break;
default: /* Unknown event */
- pr_info("Unknown WMI event type 0x%x\n",
- (int)buffer_entry[1]);
+ pr_info("Unknown WMI event type 0x%x\n", le16_to_cpu(buffer_entry[1]));
break;
}

@@ -821,7 +823,7 @@ static struct wmi_driver dell_wmi_driver = {
.id_table = dell_wmi_id_table,
.probe = dell_wmi_probe,
.remove = dell_wmi_remove,
- .notify = dell_wmi_notify,
+ .notify_new = dell_wmi_notify,
};

static int __init dell_wmi_init(void)
--
2.39.5