* [PATCH v4] ACPI: WMI: Add WMI-ACPI mapper driver
@ 2007-10-30 3:36 Carlos Corbacho
2007-10-30 13:47 ` Matthew Garrett
2007-10-30 18:18 ` Len Brown
0 siblings, 2 replies; 9+ messages in thread
From: Carlos Corbacho @ 2007-10-30 3:36 UTC (permalink / raw)
To: linux-acpi; +Cc: Len Brown
From: Carlos Corbacho <cathectic@gmail.com>
The following is an implementation of the Windows Management
Instrumentation (WMI) ACPI interface mapper (PNP0C14).
What it does:
Parses the _WDG method and exports functions to process WMI method calls,
data block query/ set commands (both based on GUID) and does basic event
handling.
i.e.
wmi_evaluate_method(const char *guid, u32 instance, u32 method_id,
const struct acpi_buffer *in, struct acpi_buffer *out)
wmi_query_block(const char *guid, u32 instance,
struct acpi_buffer *out)
wmi_set_block(const char *guid, u32 instance,
const struct acpi_buffer *in)
It also provides a helper method to find if a GUID exists or not on the
system (a quick and easy way for WMI dependant drivers to see if the
the method/ block they want exists, since GUIDs are supposed to be unique).
i.e.
bool wmi_has_guid(const char guid*)
Where guid is the text encoding of the guid, e.g.:
"67C3371D-95A3-4C37-BB61-DD47B491DAAB"
Event handling - allow a WMI based driver to register a notifier handler
with WMI. When a notification is sent to WMI, WMI will execute _WED and
then pass the results and the event to the external handler (since WMI
does not know the meaning of an event, it is left to the external drivers
to deal with, rather than WMI exporting the event to userspace).
What it might be able to do:
Handle reading data block GUIDs marked as "expensive" (e.g. calling WCxx with
the correct arguments, before and after querying the block in question). My
DSDT does not have any such marked block methods, so this is untested.
What it can't do:
Unicode - The MS article[1] calls for converting between ASCII and Unicode (or
vice versa) if a GUID is marked as "string". There is also another problem
in that given this specification revolves around Windows, Unicode in this
context likely means UTF-16. Since Linux doesn't use this, and since this
driver will only be called by other Linux drivers, it may make more sense
to treat Unicode here as UTF-8 instead.
What it won't do:
Handle a MOF[1] - the WMI mapper just implements simple calls that are
wrappers around ACPI functions. Also, MOFs are based around the Windows WMI
API, so they make little sense on Linux.
Userspace - the WMI mapper disregards the WMI-ACPI spec on this point, and
does _not_ try to export to userspace, since WMI does not exist on Linux.
All callers of the WMI mapper are assumed to be kernel space drivers.
[1] http://www.microsoft.com/whdc/system/pnppwr/wmi/wmi-acpi.mspx
===
ChangeLog
==
v1 (2007-10-02):
* Initial release
v2 (2007-10-05):
* Cleaned up code - split up super "wmi_evaluate_block" -> each external
symbol now handles its own ACPI calls, rather than handing off to
a "super" method (and in turn, is a lot simpler to read)
* Added a find_guid() symbol - return true if a given GUID exists on
the system
* wmi_* functions now return type acpi_status (since they are just
fancy wrappers around acpi_evaluate_object())
* Removed extra debug code
v3 (2007-10-27)
* More code clean up - now passes checkpatch.pl
* Change data block calls - ref MS spec, method ID is not required for
them, so drop it from the function parameters.
* Const'ify guid in the function call parameters.
* Fix _WDG buffer handling - copy the data to our own private structure.
* Change WMI from tristate to bool - otherwise the external functions are
not exported in linux/acpi.h if you try to build WMI as a module.
* Fix more flag comparisons.
* Add a maintainers entry - since I wrote this, I should take the blame
for it.
v4 (2007-10-30)
* Add missing brace from after fixing checkpatch errors.
* Rewrote event handling - allow external drivers to register with WMI to
handle WMI events
* Clean up flags and sanitise flag handling
Signed-off-by: Carlos Corbacho <cathectic@gmail.com>
---
MAINTAINERS | 7 +
drivers/acpi/Kconfig | 11 +
drivers/acpi/Makefile | 1 +
drivers/acpi/wmi.c | 590 +++++++++++++++++++++++++++++++++++++++++++++++++
include/linux/acpi.h | 17 ++
5 files changed, 626 insertions(+), 0 deletions(-)
create mode 100644 drivers/acpi/wmi.c
diff --git a/MAINTAINERS b/MAINTAINERS
index 3ceeb56..1f4fda5 100644
--- a/MAINTAINERS
+++ b/MAINTAINERS
@@ -261,6 +261,13 @@ L: linux-acpi@vger.kernel.org
W: http://acpi.sourceforge.net/
S: Supported
+ACPI WMI DRIVER
+P: Carlos Corbacho
+M: cathectic@gmail.com
+L: linux-acpi@vger.kernel.org
+W: http://www.lesswatts.org/projects/acpi/
+S: Supported
+
ADM1025 HARDWARE MONITOR DRIVER
P: Jean Delvare
M: khali@linux-fr.org
diff --git a/drivers/acpi/Kconfig b/drivers/acpi/Kconfig
index 5d0e26a..ff9cf2a 100644
--- a/drivers/acpi/Kconfig
+++ b/drivers/acpi/Kconfig
@@ -179,6 +179,17 @@ config ACPI_NUMA
depends on (X86 || IA64)
default y if IA64_GENERIC || IA64_SGI_SN2
+config ACPI_WMI
+ bool "WMI (EXPERIMENTAL)"
+ depends on EXPERIMENTAL
+ default y
+ help
+ This driver adds support for the WMI ACPI mapper device (PNP0C14)
+ found on some systems.
+
+ NOTE: You will need another driver on top of this to actually use
+ anything defined in the WMI ACPI device.
+
config ACPI_ASUS
tristate "ASUS/Medion Laptop Extras"
depends on X86
diff --git a/drivers/acpi/Makefile b/drivers/acpi/Makefile
index 54e3ab0..9e9aa39 100644
--- a/drivers/acpi/Makefile
+++ b/drivers/acpi/Makefile
@@ -55,6 +55,7 @@ obj-$(CONFIG_ACPI_THERMAL) += thermal.o
obj-$(CONFIG_ACPI_SYSTEM) += system.o event.o
obj-$(CONFIG_ACPI_DEBUG) += debug.o
obj-$(CONFIG_ACPI_NUMA) += numa.o
+obj-$(CONFIG_ACPI_WMI) += wmi.o
obj-$(CONFIG_ACPI_ASUS) += asus_acpi.o
obj-$(CONFIG_ACPI_TOSHIBA) += toshiba_acpi.o
obj-$(CONFIG_ACPI_HOTPLUG_MEMORY) += acpi_memhotplug.o
diff --git a/drivers/acpi/wmi.c b/drivers/acpi/wmi.c
new file mode 100644
index 0000000..77078fd
--- /dev/null
+++ b/drivers/acpi/wmi.c
@@ -0,0 +1,590 @@
+/*
+ * WMI to ACPI mapping driver
+ *
+ * Copyright (C) 2007 Carlos Corbacho <cathectic@gmail.com>
+ *
+ * GUID parsing code from ldm.c is:
+ * Copyright (C) 2001,2002 Richard Russon <ldm@flatcap.org>
+ * Copyright (c) 2001-2007 Anton Altaparmakov
+ * Copyright (C) 2001,2002 Jakob Kemi <jakob.kemi@telia.com>
+ *
+ * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or (at
+ * your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful, but
+ * WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License along
+ * with this program; if not, write to the Free Software Foundation, Inc.,
+ * 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
+ *
+ * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
+ *
+ * TODO:
+ *
+ * Handle method & data blocks flagged as ACPI_WMI_STRING:
+ *
+ * The MS spec says that we should translate the input from "UNICODE to ASCIIZ"
+ * and the output from "ASCIIZ to UNICODE". But by UNICODE, they probably mean
+ * UTF-16, where as we are using UTF-8/ ASCII here - so, what to do? Converting
+ * to UTF-16 is probably pointless, since most of the clients of this mapper
+ * will be Linux drivers using UTF-8/ ASCII anyway, not UTF-16.
+ *
+ * So, for the moment, the mapper should just convert input from UTF-8 to ASCII
+ */
+
+#include <linux/kernel.h>
+#include <linux/module.h>
+#include <linux/init.h>
+#include <linux/types.h>
+#include <acpi/acpi_drivers.h>
+
+#define ACPI_WMI_CLASS "wmi"
+
+#undef PREFIX
+#define PREFIX "ACPI: WMI: "
+
+MODULE_AUTHOR("Carlos Corbacho");
+MODULE_DESCRIPTION("WMI ACPI Interface Driver");
+MODULE_LICENSE("GPL");
+
+struct guid_block_t
+{
+ char guid[16];
+ union
+ {
+ char object_id[2];
+ struct
+ {
+ unsigned char notification_value;
+ unsigned char reserved;
+ };
+ };
+ u8 instance_count;
+ u8 flags;
+};
+
+struct guid_list
+{
+ struct guid_block_t *pointer;
+ int total;
+};
+
+static struct guid_list guids;
+static acpi_handle acpi_wmi_handle;
+static struct acpi_buffer *wed_buffer;
+static acpi_notify_handler wmi_external_handler;
+
+/*
+ * If the GUID data block is marked as expensive, we must enable and
+ * explicitily disable data collection.
+ */
+#define ACPI_WMI_EXPENSIVE 0x1
+#define ACPI_WMI_METHOD 0x2 /* GUID is a method */
+
+/*
+ * Data block is a string, and must be converted from ASCII to Unicode (output)
+ * or Unicode to ASCII (input)
+ */
+#define ACPI_WMI_STRING 0x4
+#define ACPI_WMI_EVENT 0x8 /* GUID is an event */
+
+static int acpi_wmi_remove(struct acpi_device *device, int type);
+static int acpi_wmi_add(struct acpi_device *device);
+
+const static struct acpi_device_id wmi_device_ids[] = {
+ {"PNP0C14", 0},
+ {"pnp0c14", 0},
+ {"", 0},
+};
+MODULE_DEVICE_TABLE(acpi, wmi_device_ids);
+
+static struct acpi_driver acpi_wmi_driver = {
+ .name = "wmi",
+ .class = ACPI_WMI_CLASS,
+ .ids = wmi_device_ids,
+ .ops = {
+ .add = acpi_wmi_add,
+ .remove = acpi_wmi_remove,
+ },
+};
+
+/*
+ * GUID parsing functions
+ */
+
+/**
+ * wmi_parse_hexbyte - Convert a ASCII hex number to a byte
+ * @src: Pointer to at least 2 characters to convert.
+ *
+ * Convert a two character ASCII hex string to a number.
+ *
+ * Return: 0-255 Success, the byte was parsed correctly
+ * -1 Error, an invalid character was supplied
+ */
+static int wmi_parse_hexbyte(const u8 *src)
+{
+ unsigned int x; /* For correct wrapping */
+ int h;
+
+ /* high part */
+ x = src[0];
+ if (x - '0' <= '9' - '0') {
+ h = x - '0';
+ } else if (x - 'a' <= 'f' - 'a') {
+ h = x - 'a' + 10;
+ } else if (x - 'A' <= 'F' - 'A') {
+ h = x - 'A' + 10;
+ } else {
+ return -1;
+ }
+ h <<= 4;
+
+ /* low part */
+ x = src[1];
+ if (x - '0' <= '9' - '0')
+ return h | (x - '0');
+ if (x - 'a' <= 'f' - 'a')
+ return h | (x - 'a' + 10);
+ if (x - 'A' <= 'F' - 'A')
+ return h | (x - 'A' + 10);
+ return -1;
+}
+
+/**
+ * wmi_swap_bytes - Rearrange GUID bytes to match GUID binary
+ * @src: Memory block holding binary GUID (16 bytes)
+ * @dest: Memory block to hold byte swapped binary GUID (16 bytes)
+ *
+ * Byte swap a binary GUID to match it's real GUID value
+ */
+static void wmi_swap_bytes(u8 *src, u8 *dest)
+{
+ int i;
+
+ for (i = 0; i <= 3; i++)
+ memcpy(dest + i, src + (3 - i), 1);
+
+ for (i = 0; i <= 1; i++)
+ memcpy(dest + 4 + i, src + (5 - i), 1);
+
+ for (i = 0; i <= 1; i++)
+ memcpy(dest + 6 + i, src + (7 - i), 1);
+
+ memcpy(dest + 8, src + 8, 8);
+}
+
+/**
+ * wmi_parse_guid - Convert GUID from ASCII to binary
+ * @src: 36 char string of the form fa50ff2b-f2e8-45de-83fa-65417f2f49ba
+ * @dest: Memory block to hold binary GUID (16 bytes)
+ *
+ * N.B. The GUID need not be NULL terminated.
+ *
+ * Return: 'true' @dest contains binary GUID
+ * 'false' @dest contents are undefined
+ */
+static bool wmi_parse_guid(const u8 *src, u8 *dest)
+{
+ static const int size[] = { 4, 2, 2, 2, 6 };
+ int i, j, v;
+
+ if (src[8] != '-' || src[13] != '-' ||
+ src[18] != '-' || src[23] != '-')
+ return false;
+
+ for (j = 0; j < 5; j++, src++) {
+ for (i = 0; i < size[j]; i++, src += 2, *dest++ = v) {
+ v = wmi_parse_hexbyte(src);
+ if (v < 0)
+ return false;
+ }
+ }
+
+ return true;
+}
+
+static bool find_guid(const char *guid_string, struct guid_block_t **out)
+{
+ char tmp[16], guid_input[16];
+ struct guid_block_t *block;
+ int i;
+
+ wmi_parse_guid(guid_string, tmp);
+ wmi_swap_bytes(tmp, guid_input);
+
+ for (i = 0; i < guids.total; i++) {
+ block = guids.pointer + i;
+
+ if (memcmp(block->guid, guid_input, 16) == 0) {
+ if (out != NULL)
+ *out = block;
+ return 1;
+ }
+ }
+ return 0;
+}
+
+static acpi_status wmi_get_event_data(u32 event)
+{
+ struct acpi_object_list input;
+ union acpi_object params[1];
+
+ input.count = 1;
+ input.pointer = params;
+ params[0].type = ACPI_TYPE_INTEGER;
+ params[0].integer.value = event;
+
+ return acpi_evaluate_object(acpi_wmi_handle, "_WED", &input,
+ wed_buffer);
+}
+
+/*
+ * Externally callable WMI functions
+ */
+/**
+ * wmi_evaluate_method - Evaluate a WMI method
+ * @guid_string: 36 char string of the form fa50ff2b-f2e8-45de-83fa-65417f2f49ba
+ * @instance: Instance index of the instance
+ * @method_id: Method ID to call
+ * &in: Buffer containing input for the method call
+ * &out: Empty buffer to return the method results
+ *
+ * Convert a WMI method call to an ACPI one, and return the results
+ */
+acpi_status wmi_evaluate_method(const char *guid_string, u32 instance,
+u32 method_id, const struct acpi_buffer *in, struct acpi_buffer *out)
+{
+ struct guid_block_t *block = NULL;
+ acpi_status status;
+ struct acpi_object_list input;
+ union acpi_object wm_params[3];
+ char method[4] = "WM";
+
+ if (!find_guid(guid_string, &block))
+ return AE_BAD_ADDRESS;
+
+ if (!block->flags & ACPI_WMI_METHOD)
+ return AE_BAD_DATA;
+
+ if (block->instance_count < instance)
+ return AE_BAD_PARAMETER;
+
+ input.count = 2;
+ input.pointer = wm_params;
+ wm_params[0].type = ACPI_TYPE_INTEGER;
+ wm_params[0].integer.value = instance;
+ wm_params[1].type = ACPI_TYPE_INTEGER;
+ wm_params[1].integer.value = method_id;
+
+ if (in != NULL) {
+ input.count = 3;
+ wm_params[2].type = ACPI_TYPE_BUFFER;
+ wm_params[2].buffer.length = in->length;
+ wm_params[2].buffer.pointer = in->pointer;
+ }
+
+ strncat(method, block->object_id, 2);
+
+ status = acpi_evaluate_object(acpi_wmi_handle, method, &input, out);
+
+ if ((block->flags & ACPI_WMI_STRING) > 0) {
+ /* Convert output from ASCIIZ to Unicode */
+ return AE_NOT_IMPLEMENTED;
+ }
+
+ return status;
+}
+EXPORT_SYMBOL(wmi_evaluate_method);
+
+/**
+ * wmi_query_block - Return contents of a WMI block
+ * @guid_string: 36 char string of the form fa50ff2b-f2e8-45de-83fa-65417f2f49ba
+ * @instance: Instance index of the instance
+ * &out: Empty buffer to return the contents of the data block to
+ *
+ * Return the contents of a data block to a buffer
+ */
+acpi_status wmi_query_block(const char *guid_string, u32 instance,
+struct acpi_buffer *out)
+{
+ struct guid_block_t *block = NULL;
+ acpi_status status;
+ struct acpi_object_list input, wc_input;
+ union acpi_object wc_params[1], wq_params[1];
+ char method[4] = "WQ";
+ char wc_method[4] = "WC";
+
+ if (guid_string == NULL || out == NULL)
+ return AE_BAD_PARAMETER;
+
+ if (!find_guid(guid_string, &block))
+ return AE_BAD_ADDRESS;
+
+ if (block->instance_count < instance)
+ return AE_BAD_PARAMETER;
+
+ /* Check GUID is a data block */
+ if ((block->flags & (ACPI_WMI_EVENT | ACPI_WMI_METHOD)) > 0)
+ return AE_BAD_ADDRESS;
+
+ input.count = 1;
+ input.pointer = wq_params;
+ wq_params[0].type = ACPI_TYPE_INTEGER;
+ wq_params[0].integer.value = instance;
+
+ /*
+ * If ACPI_WMI_EXPENSIVE, call the relevant WCxx method first to
+ * enable collection
+ */
+ if ((block->flags & ACPI_WMI_EXPENSIVE) > 0) {
+ wc_input.count = 1;
+ wc_input.pointer = wc_params;
+ wc_params[0].type = ACPI_TYPE_INTEGER;
+ wc_params[0].integer.value = 1;
+
+ strncat(wc_method, block->object_id, 2);
+
+ status = acpi_evaluate_object(acpi_wmi_handle, wc_method,
+ &wc_input, NULL);
+
+ if (ACPI_FAILURE(status))
+ return AE_ERROR;
+ }
+
+ strncat(method, block->object_id, 2);
+
+ status = acpi_evaluate_object(acpi_wmi_handle, method, NULL, out);
+
+ /*
+ * If ACPI_WMI_EXPENSIVE, call the relevant WCxx method, even if
+ * the WQxx method failed - we should disable collection anyway
+ */
+ if ((block->flags & ACPI_WMI_EXPENSIVE) > 0) {
+ wc_params[0].integer.value = 0;
+ status = acpi_evaluate_object(acpi_wmi_handle,
+ wc_method, &wc_input, NULL);
+ }
+
+ if (ACPI_SUCCESS(status)) {
+ /* Convert output from ASCIIZ to Unicode */
+ if ((block->flags & ACPI_WMI_STRING) > 0)
+ return AE_NOT_IMPLEMENTED;
+ }
+
+ return status;
+}
+EXPORT_SYMBOL(wmi_query_block);
+
+/**
+ * wmi_set_block - Write to a WMI block
+ * @guid_string: 36 char string of the form fa50ff2b-f2e8-45de-83fa-65417f2f49ba
+ * @instance: Instance index of the instance
+ * &in: Buffer containing new values for the data block
+ *
+ * Write the contents of the input buffer to ACPI
+ */
+acpi_status wmi_set_block(const char *guid_string, u32 instance,
+const struct acpi_buffer *in)
+{
+ struct guid_block_t *block = NULL;
+ struct acpi_object_list input;
+ union acpi_object params[2];
+ char method[4] = "WS";
+
+ if (guid_string == NULL || in == NULL)
+ return AE_BAD_DATA;
+
+ if (!find_guid(guid_string, &block))
+ return AE_BAD_ADDRESS;
+
+ if (block->instance_count < instance)
+ return AE_BAD_PARAMETER;
+
+ /* Check GUID is a data block */
+ if (block->flags & (ACPI_WMI_EVENT | ACPI_WMI_METHOD))
+ return AE_BAD_ADDRESS;
+
+ input.count = 2;
+ input.pointer = params;
+ params[0].type = ACPI_TYPE_INTEGER;
+ params[0].integer.value = instance;
+ params[1].type = ACPI_TYPE_BUFFER;
+ params[1].buffer.length = in->length;
+ params[1].buffer.pointer = in->pointer;
+
+ /* Convert input from Unicode to ASCIIZ */
+ if (block->flags & ACPI_WMI_STRING)
+ return AE_NOT_IMPLEMENTED;
+
+ strncat(method, block->object_id, 2);
+
+ return acpi_evaluate_object(acpi_wmi_handle, method, &input, NULL);
+}
+EXPORT_SYMBOL(wmi_set_block);
+
+/**
+ * wmi_install_notify_handler - Register handler for WMI events
+ * @handler: Function to handle notifications
+ * &data: Buffer to return data associated with event
+ *
+ * Register a handler for events sent to the WMI-ACPI mapper device.
+ */
+acpi_status wmi_install_notify_handler(acpi_notify_handler handler,
+struct acpi_buffer *data)
+{
+ if (!data)
+ return AE_BAD_PARAMETER;
+
+ if (!wed_buffer)
+ wed_buffer = data;
+ else
+ return AE_ERROR;
+
+ wmi_external_handler = handler;
+
+ return AE_OK;
+}
+EXPORT_SYMBOL(wmi_install_notify_handler);
+
+/**
+ * wmi_uninstall_notify_handler - Unregister handler for WMI events
+ *
+ * Unregister handler for events sent to the WMI-ACPI mapper device.
+ */
+acpi_status wmi_remove_notify_handler(void)
+{
+ if (wmi_external_handler) {
+ wmi_external_handler = NULL;
+ wed_buffer = NULL;
+ return AE_OK;
+ }
+ return AE_ERROR;
+}
+EXPORT_SYMBOL(wmi_remove_notify_handler);
+
+/**
+ * wmi_has_guid - Check if a GUID is available
+ * @guid_string: 36 char string of the form fa50ff2b-f2e8-45de-83fa-65417f2f49ba
+ *
+ * Check if a given GUID is defined by _WDG
+ */
+bool wmi_has_guid(const char *guid_string)
+{
+ return find_guid(guid_string, NULL);
+}
+EXPORT_SYMBOL(wmi_has_guid);
+
+/**
+ * parse_wdg - Parse the _WDG method for the GUID data blocks
+ */
+static acpi_status parse_wdg(void)
+{
+ struct acpi_buffer out = {ACPI_ALLOCATE_BUFFER, NULL};
+ union acpi_object *obj;
+ acpi_status status;
+
+ status = acpi_evaluate_object(acpi_wmi_handle, "_WDG", NULL, &out);
+
+ if (ACPI_FAILURE(status))
+ return status;
+
+ obj = (union acpi_object *) out.pointer;
+
+ if (obj->type != ACPI_TYPE_BUFFER)
+ return AE_ERROR;
+
+ guids.pointer = kzalloc(obj->buffer.length, GFP_KERNEL);
+ memcpy(guids.pointer, obj->buffer.pointer, obj->buffer.length);
+ guids.total = obj->buffer.length / sizeof(struct guid_block_t);
+
+ kfree(out.pointer);
+
+ return status;
+}
+
+static void acpi_wmi_notify(acpi_handle handle, u32 event, void *data)
+{
+ int i;
+ struct guid_block_t *block;
+
+ for (i = 0; i < guids.total; i++) {
+ block = guids.pointer + i;
+
+ if ((block->flags & ACPI_WMI_EVENT) &&
+ block->notification_value == event &&
+ wmi_external_handler && wed_buffer) {
+ /*
+ * Get data for the event, and pass it to the associated
+ * external caller
+ */
+ wmi_get_event_data(event);
+ wmi_external_handler(handle, event, wed_buffer);
+ }
+ }
+}
+
+static int acpi_wmi_add(struct acpi_device *device)
+{
+ acpi_status status;
+
+ acpi_wmi_handle = device->handle;
+
+ status = acpi_install_notify_handler(device->handle, ACPI_DEVICE_NOTIFY,
+ acpi_wmi_notify, device);
+
+ if (ACPI_FAILURE(status)) {
+ ACPI_DEBUG_PRINT((ACPI_DB_ERROR,
+ "Error installing notify handler\n"));
+ return -ENODEV;
+ }
+
+ status = parse_wdg();
+
+ if (ACPI_FAILURE(status))
+ return -ENODEV;
+
+ return 0;
+}
+
+static int acpi_wmi_remove(struct acpi_device *device, int type)
+{
+ acpi_remove_notify_handler(device->handle, ACPI_DEVICE_NOTIFY,
+ acpi_wmi_notify);
+
+ return 0;
+}
+
+static int __init acpi_wmi_init(void)
+{
+ acpi_status result;
+
+ if (acpi_disabled)
+ return -ENODEV;
+
+ result = acpi_bus_register_driver(&acpi_wmi_driver);
+
+ if (ACPI_FAILURE(result))
+ return -ENODEV;
+
+ printk(KERN_INFO PREFIX "Interface device found\n");
+ printk(KERN_INFO PREFIX "Mapper loaded\n");
+
+ return 0;
+}
+
+static void __exit acpi_wmi_exit(void)
+{
+ kfree(guids.pointer);
+ acpi_bus_unregister_driver(&acpi_wmi_driver);
+ printk(KERN_INFO PREFIX "Mapper unloaded\n");
+}
+
+module_init(acpi_wmi_init);
+module_exit(acpi_wmi_exit);
diff --git a/include/linux/acpi.h b/include/linux/acpi.h
index 8ccedf7..070d175 100644
--- a/include/linux/acpi.h
+++ b/include/linux/acpi.h
@@ -186,6 +186,23 @@ extern int ec_transaction(u8 command,
#endif /*CONFIG_ACPI_EC*/
+#ifdef CONFIG_ACPI_WMI
+
+extern acpi_status wmi_evaluate_method(const char *guid, u32 instance,
+ u32 method_id,
+ const struct acpi_buffer *in,
+ struct acpi_buffer *out);
+extern acpi_status wmi_query_block(const char *guid, u32 instance,
+ struct acpi_buffer *out);
+extern acpi_status wmi_set_block(const char *guid, u32 instance,
+ const struct acpi_buffer *in);
+extern acpi_status wmi_install_notify_handler(acpi_notify_handler handler,
+struct acpi_buffer *data);
+extern acpi_status wmi_remove_notify_handler(void);
+extern bool wmi_has_guid(const char *guid);
+
+#endif /* CONFIG_ACPI_WMI */
+
extern int acpi_blacklisted(void);
extern void acpi_bios_year(char *s);
--
1.5.3.4
^ permalink raw reply related [flat|nested] 9+ messages in thread
* Re: [PATCH v4] ACPI: WMI: Add WMI-ACPI mapper driver
2007-10-30 3:36 [PATCH v4] ACPI: WMI: Add WMI-ACPI mapper driver Carlos Corbacho
@ 2007-10-30 13:47 ` Matthew Garrett
2007-10-30 14:28 ` Carlos Corbacho
2007-10-30 18:18 ` Len Brown
1 sibling, 1 reply; 9+ messages in thread
From: Matthew Garrett @ 2007-10-30 13:47 UTC (permalink / raw)
To: Carlos Corbacho; +Cc: linux-acpi, Len Brown
On Tue, Oct 30, 2007 at 03:36:43AM +0000, Carlos Corbacho wrote:
> Event handling - allow a WMI based driver to register a notifier handler
> with WMI. When a notification is sent to WMI, WMI will execute _WED and
> then pass the results and the event to the external handler (since WMI
> does not know the meaning of an event, it is left to the external drivers
> to deal with, rather than WMI exporting the event to userspace).
I suspect that for a lot of cases, exporting this to userspace would be
helpful - it's often easier for us to do this mapping in hal and just
push out an updated fdi file than it would be to push out an updated
kernel driver.
> Userspace - the WMI mapper disregards the WMI-ACPI spec on this point, and
> does _not_ try to export to userspace, since WMI does not exist on Linux.
> All callers of the WMI mapper are assumed to be kernel space drivers.
And for similar reasons, I think it would be helpful to provide
userspace with the ability to trigger WMI calls. It's going to be easier
for people to add GUIDs to a userspace file than to extend a kernel
driver, even if there are some cases that are better handled by having a
full in-kernel driver.
--
Matthew Garrett | mjg59@srcf.ucam.org
^ permalink raw reply [flat|nested] 9+ messages in thread
* Re: [PATCH v4] ACPI: WMI: Add WMI-ACPI mapper driver
2007-10-30 13:47 ` Matthew Garrett
@ 2007-10-30 14:28 ` Carlos Corbacho
2007-10-31 10:56 ` Carlos Corbacho
0 siblings, 1 reply; 9+ messages in thread
From: Carlos Corbacho @ 2007-10-30 14:28 UTC (permalink / raw)
To: Matthew Garrett; +Cc: linux-acpi, Len Brown
On Tuesday 30 Otober 2007 13:47:01 Matthew Garrett wrote:
> I suspect that for a lot of cases, exporting this to userspace would be
> helpful - it's often easier for us to do this mapping in hal and just
> push out an updated fdi file than it would be to push out an updated
> kernel driver.
But what would be mapped via HAL?
WMI is such a completely undefined "standard", that not only do you have to map
the GUIDs, but you'd also have to map the program logic as well (marshalling
arguments, restricting function parameters, call order, etc) and this strikes
me as being beyond the scope of an fdi file.
For some methods, which are quite simple, this _might_ be possible. For others
I've seen, this will end up a complete mess.
If you're going to do all that, you may as well write a driver in kernel space
and leave that to export the functionality in a way userspace can already handle.
> And for similar reasons, I think it would be helpful to provide
> userspace with the ability to trigger WMI calls. It's going to be easier
> for people to add GUIDs to a userspace file than to extend a kernel
> driver, even if there are some cases that are better handled by having a
> full in-kernel driver.
There are also other technical problems with this, beyond my own personal
disagreement - since WMI relies so heavily on ACPI (well, the driver here is
really just a simple wrapper), then you would have to start exporting bits of ACPI
to userspace to access WMI. As mentioned in the original patches - the mapper knows
nothing about "types", this knowledge is known only by the caller - WMI just deals
in acpi_buffer. My understanding is that ACPI would have to export the required
functions and data structures so that userspace could provide then translate from
acpi_buffer to the known type (unless there is another way round this to import and
export the data to/ from userspace?)
-Carlos
--
E-Mail: cathectic@gmail.com
Web: strangeworlds.co.uk
GPG Key ID: 0x23EE722D
^ permalink raw reply [flat|nested] 9+ messages in thread
* Re: [PATCH v4] ACPI: WMI: Add WMI-ACPI mapper driver
2007-10-30 3:36 [PATCH v4] ACPI: WMI: Add WMI-ACPI mapper driver Carlos Corbacho
2007-10-30 13:47 ` Matthew Garrett
@ 2007-10-30 18:18 ` Len Brown
2007-10-30 22:52 ` Carlos Corbacho
1 sibling, 1 reply; 9+ messages in thread
From: Len Brown @ 2007-10-30 18:18 UTC (permalink / raw)
To: Carlos Corbacho; +Cc: linux-acpi, Nashif, Anas
When I consulted Anas about WMI, he recommended that Linux
expose WMI via CIMOM. I think that this means we'd need
to invent a sysfs interface for this acpi->wmi driver
to expose the hooks to a user-space daemon, which
would then make sense of it in Linux's management framework.
Anas provided the references below.
thanks,
-Len
http://developer.novell.com/wiki/index.php/OMC
This is a Novell project with many providers dealing will many aspects
of the OS. They also provide RPMs for OpenSUSE and is the main source
for the OpenWBEM CIMOM binaries for SUSE.
http://sblim.wiki.sourceforge.net/
Is the IBM project dealing with instrumentation and home of the SFCB
CIMOM targeting embedded environments and only supports CMPI providers.
http://sblim.wiki.sourceforge.net/ProviderCmpiSysfs is a sysfs provider.
Seems to be old but is a good example.
http://openpegasus.org/
This is the home of the OpenPeagsus CIMOM, supported by multiple
vendors.
http://www.dmtf.org/standards/wbem/ is about the WBEM protocol. Also the
tutorial is good: http://www.wbemsolutions.com/tutorials/CIM/wbem.html
http://cimple.org/ tries to make writing providers easy.
^ permalink raw reply [flat|nested] 9+ messages in thread
* Re: [PATCH v4] ACPI: WMI: Add WMI-ACPI mapper driver
2007-10-30 18:18 ` Len Brown
@ 2007-10-30 22:52 ` Carlos Corbacho
2007-10-31 0:18 ` Carlos Corbacho
0 siblings, 1 reply; 9+ messages in thread
From: Carlos Corbacho @ 2007-10-30 22:52 UTC (permalink / raw)
To: Len Brown; +Cc: linux-acpi, Nashif, Anas
On Tuesday 30 October 2007 18:18:22 Len Brown wrote:
> When I consulted Anas about WMI, he recommended that Linux
> expose WMI via CIMOM. I think that this means we'd need
> to invent a sysfs interface for this acpi->wmi driver
> to expose the hooks to a user-space daemon, which
> would then make sense of it in Linux's management framework.
The main problem to overcome with a sysfs interface is that a
WMI-ACPI call takes multiple values (worst case scenario is a
method, which takes three arguments - method id, instance and
an input buffer).
We could try and do something like this for methods & data blocks:
/sys/firmware/acpi/wmi/<GUID>/ ->
type (method, event, data)
instance_count (maximum number of instances - method's & data only)
Then have the following directory for each possible instance:
/sys/firmware/acpi/wmi/<GUID>/<instance>/
Inside, we'd have two files:
method_id (methods only)
data
For data blocks, it's easy -> just read/ write from data (wmi.c could
also try to be clever, and set the correct read/ write permissions
by checking for appropriate methods).
For method calling -> This is more complicated, since we need to pass
a method id and the input data (and unlike instance, WMI-ACPI does not
define the maximum number of methods). We could do this by writing to
a file called method_id first with the method_id we want to call, then
read/ write to data (but I'm open to suggestions for a better interface).
Events would, I imagine, be some sort of polling, although probably
split over two files:
One file to tell the polling application of the event, another file
(say 'data'), containing the data associated with that event?
Unless there's another way we can export the events without polling?
-Carlos
--
E-Mail: cathectic@gmail.com
Web: strangeworlds.co.uk
GPG Key ID: 0x23EE722D
^ permalink raw reply [flat|nested] 9+ messages in thread
* Re: [PATCH v4] ACPI: WMI: Add WMI-ACPI mapper driver
2007-10-30 22:52 ` Carlos Corbacho
@ 2007-10-31 0:18 ` Carlos Corbacho
2007-10-31 12:21 ` Carlos Corbacho
0 siblings, 1 reply; 9+ messages in thread
From: Carlos Corbacho @ 2007-10-31 0:18 UTC (permalink / raw)
To: Len Brown; +Cc: linux-acpi, Nashif, Anas
On Tuesday 30 October 2007 22:52:30 Carlos Corbacho wrote:
> On Tuesday 30 October 2007 18:18:22 Len Brown wrote:
> > When I consulted Anas about WMI, he recommended that Linux
> > expose WMI via CIMOM. I think that this means we'd need
> > to invent a sysfs interface for this acpi->wmi driver
> > to expose the hooks to a user-space daemon, which
> > would then make sense of it in Linux's management framework.
>
> The main problem to overcome with a sysfs interface is that a
> WMI-ACPI call takes multiple values (worst case scenario is a
> method, which takes three arguments - method id, instance and
> an input buffer).
Also, for input, I don't know how to handle the size/ type problem[1] - only
this user-space daemon would know what size/ type of buffer the input
requires (via the relevant MOF in the management framework) - this
information would also need to be conveyed through the sysfs interface to the
mapper somehow.
e.g. If we write '1' (assume we have already set the method id somehow)
to /sys/firmware/acpi/wmi/<guid>/<instance>/data
What type/ size is the correct size of the buffer parameter to be passed to
ACPI - u8 (1 byte), u16 (2 bytes), u32 (4 bytes), etc?
So, thinking this over more carefully, whilst sysfs can probably be used to
export data, I'm not entirely convinced for using it to input arbitrary data
types, unless there's some way we can get round this?
-Carlos
[1] For a buffer to be passed to acpi_evaluate_object() as a parameter, we
must know the size in bytes to use beforehand when we create the buffer -
otherwise, if we pass an incorrectly sized buffer as a parameter, the
acpi_evaluate_object() call will fail. However, WMI-ACPI does not specify
this - so this information is known only to the MOF writer, or caller of the
WMI method/ data set.
--
E-Mail: cathectic@gmail.com
Web: strangeworlds.co.uk
GPG Key ID: 0x23EE722D
^ permalink raw reply [flat|nested] 9+ messages in thread
* Re: [PATCH v4] ACPI: WMI: Add WMI-ACPI mapper driver
2007-10-30 14:28 ` Carlos Corbacho
@ 2007-10-31 10:56 ` Carlos Corbacho
0 siblings, 0 replies; 9+ messages in thread
From: Carlos Corbacho @ 2007-10-31 10:56 UTC (permalink / raw)
To: Matthew Garrett; +Cc: linux-acpi, Len Brown
On Tuesday 30 October 2007 14:28:10 Carlos Corbacho wrote:
> There are also other technical problems with this, beyond my own personal
> disagreement - since WMI relies so heavily on ACPI (well, the driver here
> is really just a simple wrapper), then you would have to start exporting
> bits of ACPI to userspace to access WMI. As mentioned in the original
> patches - the mapper knows nothing about "types", this knowledge is known
> only by the caller - WMI just deals in acpi_buffer. My understanding is
> that ACPI would have to export the required functions and data structures
> so that userspace could provide then translate from acpi_buffer to the
> known type (unless there is another way round this to import and export the
> data to/ from userspace?)
To backtrack slightly - I'm wrong on this point; after sleeping on this for
the night (and/ or application of cluebat), I can modify the functions to let
userspace call them without exporting half of ACPI in the process.
However, I still believe that HAL is the wrong application to deal with WMI.
-Carlos
--
E-Mail: cathectic@gmail.com
Web: strangeworlds.co.uk
GPG Key ID: 0x23EE722D
^ permalink raw reply [flat|nested] 9+ messages in thread
* Re: [PATCH v4] ACPI: WMI: Add WMI-ACPI mapper driver
2007-10-31 0:18 ` Carlos Corbacho
@ 2007-10-31 12:21 ` Carlos Corbacho
2007-10-31 16:24 ` Carlos Corbacho
0 siblings, 1 reply; 9+ messages in thread
From: Carlos Corbacho @ 2007-10-31 12:21 UTC (permalink / raw)
To: Len Brown; +Cc: linux-acpi, Nashif, Anas
On Wednesday 31 October 2007 00:18:19 Carlos Corbacho wrote:
> On Tuesday 30 October 2007 22:52:30 Carlos Corbacho wrote:
> > On Tuesday 30 October 2007 18:18:22 Len Brown wrote:
> > > When I consulted Anas about WMI, he recommended that Linux
> > > expose WMI via CIMOM. I think that this means we'd need
> > > to invent a sysfs interface for this acpi->wmi driver
> > > to expose the hooks to a user-space daemon, which
> > > would then make sense of it in Linux's management framework.
> >
> > The main problem to overcome with a sysfs interface is that a
> > WMI-ACPI call takes multiple values.
>
> So, thinking this over more carefully, whilst sysfs can probably be used to
> export data, I'm not entirely convinced for using it to input arbitrary
> data types, unless there's some way we can get round this?
After sleeping on the matter, I'm now certain that sysfs is the wrong way to
go for allowing userspace to access WMI-ACPI - we're trying to pass and
return too much data for sysfs to handle.
So, what we can do is modify the proposed internal kernel calls for userspace
(since a little sleep is a wonderful thing) to something like:
wmi_evaluate_method(const char* guid, __u32 instance, __u32 method id, void
*input, __u32 input_size, void *output, __u32 output_size)
(wmi_evaluate_method could be shrunk to just _one_ buffer, since the WMI-ACPI
spec suggests the mapper should only take & return one buffer).
wmi_query_data(const char* guid, __u32 instance, void *output, __u32
output_size)
wmi_set_data(const char* guid, __u32 instance, void *input, __u32 input_size)
(Event handling is a little more tricky - we could create a
new 'wmi_event_handler', and allow userspace or in kernel drivers to register
this handler with the WMI-ACPI mapper, and avoid polling).
Then, we leave it up to OpenWBEM, etc to write their own backend to handle
WMI-ACPI (I'm not sure if "provider" is the right terminology to use with
WMI-ACPI, since we're just converting WMI calls to ACPI). This backend would
take a WMI-ACPI method call, and convert that to the correct kernel function
call (the kernel function then handles making the ACPI call and returning the
results). Interpretation is left to the backend and its associated MOF.
(For the case of WMI not being built in, we define all the functions to return
an error, or if WMI is enabled but there is no WMI-ACPI mapping device, wmi.c
would return the errors instead).
Would this be an acceptable solution? (Len - If this is the case, where do we
expose the wmi_* functions to userspace? linux/acpi.h, or create a new
include file?)
-Carlos
--
E-Mail: cathectic@gmail.com
Web: strangeworlds.co.uk
GPG Key ID: 0x23EE722D
^ permalink raw reply [flat|nested] 9+ messages in thread
* Re: [PATCH v4] ACPI: WMI: Add WMI-ACPI mapper driver
2007-10-31 12:21 ` Carlos Corbacho
@ 2007-10-31 16:24 ` Carlos Corbacho
0 siblings, 0 replies; 9+ messages in thread
From: Carlos Corbacho @ 2007-10-31 16:24 UTC (permalink / raw)
To: Len Brown; +Cc: linux-acpi, Nashif, Anas
On Wednesday 31 October 2007 12:21:10 Carlos Corbacho wrote:
> On Wednesday 31 October 2007 00:18:19 Carlos Corbacho wrote:
> > On Tuesday 30 October 2007 22:52:30 Carlos Corbacho wrote:
> > > On Tuesday 30 October 2007 18:18:22 Len Brown wrote:
> > > > When I consulted Anas about WMI, he recommended that Linux
> > > > expose WMI via CIMOM. I think that this means we'd need
> > > > to invent a sysfs interface for this acpi->wmi driver
> > > > to expose the hooks to a user-space daemon, which
> > > > would then make sense of it in Linux's management framework.
> > >
> > > The main problem to overcome with a sysfs interface is that a
> > > WMI-ACPI call takes multiple values.
> >
> > So, thinking this over more carefully, whilst sysfs can probably be used
> > to export data, I'm not entirely convinced for using it to input
> > arbitrary data types, unless there's some way we can get round this?
>
> After sleeping on the matter, I'm now certain that sysfs is the wrong way
> to go for allowing userspace to access WMI-ACPI - we're trying to pass and
> return too much data for sysfs to handle.
On second thoughts, my other proposal won't work either - we'd end up with a
mess of either ioctl's or sysctl's to call from userspace (I really should
read up on my userspace interaction a bit more).
The sysfs proposal could be made to work, it will just be a little cumbersome
(I'll have a go at implementing it and see how it works out).
-Carlos
--
E-Mail: cathectic@gmail.com
Web: strangeworlds.co.uk
GPG Key ID: 0x23EE722D
^ permalink raw reply [flat|nested] 9+ messages in thread
end of thread, other threads:[~2007-10-31 16:23 UTC | newest]
Thread overview: 9+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2007-10-30 3:36 [PATCH v4] ACPI: WMI: Add WMI-ACPI mapper driver Carlos Corbacho
2007-10-30 13:47 ` Matthew Garrett
2007-10-30 14:28 ` Carlos Corbacho
2007-10-31 10:56 ` Carlos Corbacho
2007-10-30 18:18 ` Len Brown
2007-10-30 22:52 ` Carlos Corbacho
2007-10-31 0:18 ` Carlos Corbacho
2007-10-31 12:21 ` Carlos Corbacho
2007-10-31 16:24 ` Carlos Corbacho
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox