[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <4AE20560.5020908@gmail.com>
Date: Fri, 23 Oct 2009 23:34:56 +0400
From: Alexey Starikovskiy <aystarik@...il.com>
To: Ike Panhc <ike.pan@...onical.com>
CC: linux-acpi@...r.kernel.org, linux-kernel@...r.kernel.org,
Alexandre Rostovtsev <tetromino@...il.com>
Subject: Re: [Resend] [PATCH] ACPI: New driver for Lenovo SL laptops
Hi,
Could you please post acpidump from this machine, I would like to
understand,
why you need to register for EC query events.
Thanks,
Alex.
Ike Panhc пишет:
> lenovo-sl-laptop is a new driver that provides support for hotkeys, bluetooth,
> LenovoCare LEDs and fan speed on the Lenovo ThinkPad SL series laptops. The
> original author is Alexandre Rostovtsev. [1] In February 2009 Alexandre has
> posted the driver on the linux-acpi mailing list and and there was some
> feedback suggesting further modifications. [2] I would like to see Linux
> working properly on these laptops. I was encouraged to push this driver again
> with the modifications that where suggested in the responses to the initial
> post in order to allow me and others interested in that driver to improve it
> and hopefully get it included upstream.
>
> [1] homepage : http://github.com/tetromino/lenovo-sl-laptop/tree/master
> [2] http://patchwork.kernel.org/patch/7427/
>
> Following the suggestions when last time the origin author has posted on the
> linux-acpi mailing list. The major modification of this driver is listed below.
> - Remove backlight control
> - Remove procfs EC debug
> - Remove fan control function
> - Using generic debugging infrastructure
> - Support for lastest rfkill infrastructure (by Alexandre)
> - Register query function into EC for detecting hotkey event
>
> Patch against current checkout of linux-acpi 2.6.31 is below.
>
> The major modification of this driver since last time I posted this driver on
> linux-acpi mailing list [3] is listed below.
> - Dont free input device when exit
> - Not to register rfkill on a device which not exist
> - Add the poll function on rfkill, I found a way to register a notify function
> on H/W radio switch (on EC event with query bit 0x81), but after register,
> the H/W radio switch is no longer function to turn off the radio device. I
> will keep finding a way to use this event.
> - Remove auto_enable parameters of rfkill since we have a hardware switch.
> - Using hotkey deivce ids for module alias
> - Let the driver simpler for reading
>
> [3] http://patchwork.kernel.org/patch/49912/
>
> === 8< ===
>
> lenovo-sl-laptop: Extra driver for Lenovo SL series laptop
>
> This driver provides support for the following functions.
> - Hotkeys: LenovoCare, Volumn up/down/mute, Battery, Suspend, WLAN switch,
> Video switch, Pointer switch (as KEY_PROG1), Dock eject (as
> KEY_PROG2), Hibernate, Lock screen, Screen Zoom and LCD brightness
> up/down.
> - Radio RFKILL: switching on/off UWB, bluetooth and wifi.
> - LenovoCare LEDs: On, off, Dimmed blinking and standard blinking.
> (Blinking supported with ledtrig_timer)
> - Fan speed: Reading current fan speed
>
> The original author of this driver is Alexandre Rostovtsev
>
> The Lenovo ThinkPad SL series laptops are not supported by the normal
> thinkpad_acpi driver because their firmware is quite different from the
> T-series/R-series/X-series ThinkPads. [3]
>
> [3] http://mailman.linux-thinkpad.org/pipermail/linux-thinkpad/2009-January/046122.html
>
>
> Signed-off-by: Ike Panhc <ike.pan@...onical.com>
>
> ---
> drivers/platform/x86/Kconfig | 12 +
> drivers/platform/x86/Makefile | 1 +
> drivers/platform/x86/lenovo-sl-laptop.c | 721 +++++++++++++++++++++++++++++++
> 3 files changed, 734 insertions(+), 0 deletions(-)
> create mode 100644 drivers/platform/x86/lenovo-sl-laptop.c
>
> diff --git a/drivers/platform/x86/Kconfig b/drivers/platform/x86/Kconfig
> index 55ca39d..1ae72e3 100644
> --- a/drivers/platform/x86/Kconfig
> +++ b/drivers/platform/x86/Kconfig
> @@ -143,6 +143,18 @@ config HP_WMI
> To compile this driver as a module, choose M here: the module will
> be called hp-wmi.
>
> +config LENOVO_SL_LAPTOP
> + tristate "Lenovo ThinkPad SL Series Laptop Extras"
> + depends on ACPI
> + select HWMON
> + select INPUT
> + select RFKILL
> + ---help---
> + This is a driver for the Lenovo ThinkPad SL series laptops
> + (SL300/400/500), which are not supported by the thinkpad_acpi
> + driver. This driver adds support for hotkeys, rfkill control,
> + the Lenovo Care LED, fan speed.
> +
> config MSI_LAPTOP
> tristate "MSI Laptop Extras"
> depends on ACPI
> diff --git a/drivers/platform/x86/Makefile b/drivers/platform/x86/Makefile
> index d1c1621..1037739 100644
> --- a/drivers/platform/x86/Makefile
> +++ b/drivers/platform/x86/Makefile
> @@ -11,6 +11,7 @@ obj-$(CONFIG_DELL_WMI) += dell-wmi.o
> obj-$(CONFIG_ACER_WMI) += acer-wmi.o
> obj-$(CONFIG_ACERHDF) += acerhdf.o
> obj-$(CONFIG_HP_WMI) += hp-wmi.o
> +obj-$(CONFIG_LENOVO_SL_LAPTOP) += lenovo-sl-laptop.o
> obj-$(CONFIG_TC1100_WMI) += tc1100-wmi.o
> obj-$(CONFIG_SONY_LAPTOP) += sony-laptop.o
> obj-$(CONFIG_THINKPAD_ACPI) += thinkpad_acpi.o
> diff --git a/drivers/platform/x86/lenovo-sl-laptop.c b/drivers/platform/x86/lenovo-sl-laptop.c
> new file mode 100644
> index 0000000..d8fc093
> --- /dev/null
> +++ b/drivers/platform/x86/lenovo-sl-laptop.c
> @@ -0,0 +1,721 @@
> +/*
> + * lenovo-sl-laptop.c - Lenovo ThinkPad SL Series Extras Driver
> + *
> + *
> + * Copyright (C) 2008-2009 Alexandre Rostovtsev <tetromino@...il.com>
> + * 2009 Ike Panhc <ike.pan@...onical.com>
> + *
> + * Largely based on thinkpad_acpi.c, eeepc-laptop.c, and video.c which
> + * are copyright their respective authors.
> + *
> + * The original website of this driver is at
> + * http://github.com/tetromino/lenovo-sl-laptop/tree/master
> + *
> + * 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., 51 Franklin Street, Fifth Floor, Boston, MA
> + * 02110-1301, USA.
> + *
> + */
> +#include <linux/module.h>
> +#include <linux/kernel.h>
> +#include <linux/version.h>
> +#include <linux/init.h>
> +#include <linux/acpi.h>
> +#include <linux/pci_ids.h>
> +#include <linux/rfkill.h>
> +#include <linux/hwmon.h>
> +#include <linux/hwmon-sysfs.h>
> +#include <linux/platform_device.h>
> +#include <linux/input.h>
> +#include <linux/uaccess.h>
> +
> +#define LENOVO_SL_MODULE_DESC "Lenovo ThinkPad SL Series Extras driver"
> +#define LENOVO_SL_MODULE_NAME "lenovo-sl-laptop"
> +#define ACPI_EC0_PATH "\\_SB.PCI0.SBRG.EC0"
> +#define ACPI_HKEY_PATH ACPI_EC0_PATH ".HKEY"
> +#define LENOVO_SL_MAX_ACPI_ARGS 3
> +
> +MODULE_AUTHOR("Alexandre Rostovtsev");
> +MODULE_AUTHOR("Ike Panhc");
> +MODULE_DESCRIPTION(LENOVO_SL_MODULE_DESC);
> +MODULE_LICENSE("GPL");
> +
> +/* general */
> +
> +static acpi_handle lenovo_sl_laptop_hkey_handle;
> +static acpi_handle lenovo_sl_laptop_ec0_handle;
> +static struct platform_device *lenovo_sl_laptop_pdev;
> +
> +static int lensl_acpi_int_func(acpi_handle handle, char *pathname,
> + int *ret, int n_arg, ...)
> +{
> + acpi_status status;
> + struct acpi_object_list params;
> + union acpi_object in_obj[LENOVO_SL_MAX_ACPI_ARGS], out_obj;
> + struct acpi_buffer result, *resultp;
> + int i;
> + va_list ap;
> +
> + if (!handle)
> + return -EINVAL;
> + if (n_arg < 0 || n_arg > LENOVO_SL_MAX_ACPI_ARGS)
> + return -EINVAL;
> + va_start(ap, n_arg);
> + for (i = 0; i < n_arg; i++) {
> + in_obj[i].integer.value = va_arg(ap, int);
> + in_obj[i].type = ACPI_TYPE_INTEGER;
> + }
> + va_end(ap);
> + params.count = n_arg;
> + params.pointer = in_obj;
> +
> + if (ret) {
> + result.length = sizeof(out_obj);
> + result.pointer = &out_obj;
> + resultp = &result;
> + } else
> + resultp = NULL;
> +
> + status = acpi_evaluate_object(handle, pathname, ¶ms, resultp);
> + if (ACPI_FAILURE(status))
> + return -EIO;
> + if (ret)
> + *ret = out_obj.integer.value;
> +
> + return 0;
> +}
> +
> +/*************************************************************************
> + Bluetooth, WWAN, UWB
> + *************************************************************************/
> +
> +/* ACPI GBDC/SBDC, GWAN/SWAN, GUWB/SUWB bits */
> +#define LENOVO_SL_RADIO_HWPRESENT (0x01) /* hardware is available */
> +#define LENOVO_SL_RADIO_RADIOSSW (0x02) /* radio is enabled */
> +#define LENOVO_SL_RADIO_RESUMECTRL (0x04) /* state at resume: off/last state */
> +
> +struct lensl_radio {
> + int type;
> + enum rfkill_type rfktype;
> + char *rfkname;
> + struct rfkill *rfk;
> + char *get_pathname;
> + char *set_pathname;
> +};
> +
> +static int radio_get_acpi(char *pathname, int *value)
> +{
> + return lensl_acpi_int_func(lenovo_sl_laptop_hkey_handle, pathname,
> + value, 0);
> +}
> +
> +static int radio_set_acpi(char *pathname, int value)
> +{
> + return lensl_acpi_int_func(lenovo_sl_laptop_hkey_handle, pathname,
> + NULL, 1, value);
> +}
> +
> +static int radio_get(struct lensl_radio *radio, bool *sw_blocked,
> + bool *hw_blocked)
> +{
> + int wlsw;
> + int value;
> +
> + if (!radio)
> + return -EINVAL;
> + if (!radio_get_acpi("WLSW", &wlsw) && wlsw)
> + *hw_blocked = 0;
> + else
> + *hw_blocked = 1;
> + if (radio_get_acpi(radio->get_pathname, &value))
> + return -ENODEV;
> + if (!(value & LENOVO_SL_RADIO_HWPRESENT))
> + return -ENODEV;
> + if (value & LENOVO_SL_RADIO_RADIOSSW)
> + *sw_blocked = 0;
> + else
> + *sw_blocked = 1;
> + return 0;
> +}
> +
> +static int radio_set(struct lensl_radio *radio, bool blocked)
> +{
> + int res, value;
> +
> + res = radio_get_acpi(radio->get_pathname, &value);
> + if (res)
> + return res;
> +
> + if (blocked)
> + value &= ~LENOVO_SL_RADIO_RADIOSSW;
> + else
> + value |= LENOVO_SL_RADIO_RADIOSSW;
> + if (radio_set_acpi(radio->set_pathname, value))
> + return -EIO;
> +
> + return 0;
> +}
> +
> +/* Bluetooth/WWAN/UWB rfkill interface */
> +
> +static void radio_rfkill_query(struct rfkill *rfk, void *data)
> +{
> + struct lensl_radio *radio = data;
> + int res;
> + bool sw_blocked, hw_blocked;
> +
> + if (!radio)
> + return;
> +
> + res = radio_get(radio, &sw_blocked, &hw_blocked);
> + if (res)
> + return;
> +
> + rfkill_set_states(rfk, sw_blocked, hw_blocked);
> +}
> +
> +static int radio_rfkill_set_block(void *data, bool blocked)
> +{
> + struct lensl_radio *radio = data;
> + int res;
> + bool sw_blocked, hw_blocked;
> +
> + if (!radio)
> + return -EINVAL;
> +
> + res = radio_get(radio, &sw_blocked, &hw_blocked);
> + if (res)
> + return res;
> +
> + if (hw_blocked)
> + return 0;
> + if (sw_blocked == blocked)
> + return 0;
> +
> + return radio_set(radio, sw_blocked);
> +}
> +
> +static struct rfkill_ops radio_rfkops = {
> + .poll = radio_rfkill_query,
> + .query = radio_rfkill_query,
> + .set_block = radio_rfkill_set_block,
> +};
> +
> +/* Bluetooth/WWAN/UWB init and exit and HW switch notification */
> +
> +static struct lensl_radio radio_radios[3] = {
> +#define RADIO_BLUETOOTH (0)
> + {
> + .type = RADIO_BLUETOOTH,
> + .rfktype = RFKILL_TYPE_BLUETOOTH,
> + .rfkname = "lenovo-sl-bluetooth",
> + .get_pathname = "GBDC",
> + .set_pathname = "SBDC",
> + },
> +#define RADIO_WWAN (1)
> + {
> + .type = RADIO_WWAN,
> + .rfktype = RFKILL_TYPE_WWAN,
> + .rfkname = "lenovo-sl-wwan",
> + .get_pathname = "GWAN",
> + .set_pathname = "SWAN",
> + },
> +#define RADIO_UWB (2)
> + {
> + .type = RADIO_UWB,
> + .rfktype = RFKILL_TYPE_UWB,
> + .rfkname = "lenovo-sl-uwb",
> + .get_pathname = "GUWB",
> + .set_pathname = "SUWB",
> + },
> +};
> +
> +static void radio_exit(int type)
> +{
> + if (radio_radios[type].rfk) {
> + rfkill_unregister(radio_radios[type].rfk);
> + rfkill_destroy(radio_radios[type].rfk);
> + radio_radios[type].rfk = NULL;
> + }
> +}
> +
> +static int radio_init(int type)
> +{
> + int res;
> + bool sw_blocked, hw_blocked;
> +
> + if (!lenovo_sl_laptop_hkey_handle)
> + return -ENODEV;
> +
> + /* 1st: Get the sw/hw status */
> + res = radio_get(&radio_radios[type], &sw_blocked, &hw_blocked);
> + if (res)
> + return res;
> +
> + /* 2nd: allocate rfkill */
> + radio_radios[type].rfk = rfkill_alloc(radio_radios[type].rfkname,
> + &lenovo_sl_laptop_pdev->dev,
> + radio_radios[type].rfktype,
> + &radio_rfkops,
> + &radio_radios[type]);
> + if (!(radio_radios[type].rfk)) {
> + pr_err("Failed to allocate memory for rfkill class\n");
> + return -ENOMEM;
> + }
> +
> + /* 3rd: Set status */
> + rfkill_init_sw_state(radio_radios[type].rfk, sw_blocked);
> + rfkill_set_hw_state(radio_radios[type].rfk, hw_blocked);
> +
> + /* 4th: Register rfkill */
> + res = rfkill_register(radio_radios[type].rfk);
> + if (res < 0) {
> + pr_err("Failed to register %s rfkill switch: %d\n",
> + radio_radios[type].rfkname, res);
> + rfkill_destroy(radio_radios[type].rfk);
> + radio_radios[type].rfk = NULL;
> + }
> +
> + return res;
> +}
> +
> +/*************************************************************************
> + LEDs
> + *************************************************************************/
> +#ifdef CONFIG_NEW_LEDS
> +
> +#define LED_OFF 0
> +#define LED_ON 0x02
> +#define LED_BLINK 0x01
> +#define LED_DIM 0x100
> +
> +/* equivalent to the ThinkVantage LED on other ThinkPads */
> +#define LED_NAME "lensl::lenovocare"
> +#define LED_WQ_NAME "lenovo-sl-led-wq"
> +
> +static struct workqueue_struct *led_wq;
> +
> +struct {
> + struct led_classdev cdev;
> + enum led_brightness brightness;
> + int supported, new_code;
> + struct work_struct work;
> +} led_tv;
> +
> +static inline int led_set_tvls(int code)
> +{
> + return lensl_acpi_int_func(lenovo_sl_laptop_hkey_handle, "TVLS", NULL,
> + 1, code);
> +}
> +
> +static void led_tv_worker(struct work_struct *work)
> +{
> + if (!led_tv.supported)
> + return;
> + led_set_tvls(led_tv.new_code);
> + if (led_tv.new_code)
> + led_tv.brightness = LED_FULL;
> + else
> + led_tv.brightness = LED_OFF;
> +}
> +
> +static void led_tv_brightness_set_sysfs(struct led_classdev *led_cdev,
> + enum led_brightness brightness)
> +{
> + switch (brightness) {
> + case LED_OFF:
> + led_tv.new_code = LED_OFF;
> + break;
> + case LED_FULL:
> + led_tv.new_code = LED_ON;
> + break;
> + default:
> + return;
> + }
> + queue_work(led_wq, &led_tv.work);
> +}
> +
> +static enum led_brightness led_tv_brightness_get_sysfs(
> + struct led_classdev *led_cdev)
> +{
> + return led_tv.brightness;
> +}
> +
> +static int led_tv_blink_set_sysfs(struct led_classdev *led_cdev,
> + unsigned long *delay_on,
> + unsigned long *delay_off)
> +{
> + if (*delay_on == 0 && *delay_off == 0) {
> + /* If we can choose the flash rate, use dimmed blinking --
> + it looks better */
> + led_tv.new_code = LED_ON |
> + LED_BLINK | LED_DIM;
> + *delay_on = 2000;
> + *delay_off = 2000;
> + } else if (*delay_on + *delay_off == 4000) {
> + /* User wants dimmed blinking */
> + led_tv.new_code = LED_ON |
> + LED_BLINK | LED_DIM;
> + } else if (*delay_on == 7250 && *delay_off == 500) {
> + /* User wants standard blinking mode */
> + led_tv.new_code = LED_ON | LED_BLINK;
> + } else
> + return -EINVAL;
> + queue_work(led_wq, &led_tv.work);
> + return 0;
> +}
> +
> +static void led_exit(void)
> +{
> + led_set_tvls(LED_OFF);
> + destroy_workqueue(led_wq);
> + if (led_tv.supported) {
> + led_classdev_unregister(&led_tv.cdev);
> + led_tv.supported = 0;
> + }
> +}
> +
> +static int led_init(void)
> +{
> + int res;
> +
> + led_wq = create_singlethread_workqueue(LED_WQ_NAME);
> + if (!led_wq) {
> + pr_err("Failed to create a workqueue\n");
> + return -ENOMEM;
> + }
> +
> + memset(&led_tv, 0, sizeof(led_tv));
> + led_tv.cdev.brightness_get = led_tv_brightness_get_sysfs;
> + led_tv.cdev.brightness_set = led_tv_brightness_set_sysfs;
> + led_tv.cdev.blink_set = led_tv_blink_set_sysfs;
> + led_tv.cdev.name = LED_NAME;
> + INIT_WORK(&led_tv.work, led_tv_worker);
> + led_set_tvls(LED_ON);
> + res = led_classdev_register(&lenovo_sl_laptop_pdev->dev, &led_tv.cdev);
> + if (res) {
> + pr_warning("Failed to register LED device\n");
> + return res;
> + }
> + led_tv.supported = 1;
> + return 0;
> +}
> +
> +#else /* CONFIG_NEW_LEDS */
> +
> +static void led_exit(void)
> +{
> +}
> +
> +static int led_init(void)
> +{
> + return -ENODEV;
> +}
> +
> +#endif /* CONFIG_NEW_LEDS */
> +
> +/*************************************************************************
> + hwmon & fans
> + *************************************************************************/
> +
> +static struct device *hwmon_device;
> +
> +static inline int hwmon_get_tach(int *value, int fan)
> +{
> + return lensl_acpi_int_func(lenovo_sl_laptop_ec0_handle, "TACH", value,
> + 1, fan);
> +}
> +
> +static ssize_t hwmon_fan1_input_show(struct device *dev,
> + struct device_attribute *attr, char *buf)
> +{
> + int res;
> + int rpm;
> +
> + res = hwmon_get_tach(&rpm, 0);
> + if (res)
> + return res;
> + return snprintf(buf, PAGE_SIZE, "%u\n", rpm);
> +}
> +
> +static struct device_attribute hwmon_fan1_input =
> + __ATTR(fan1_input, S_IRUGO, hwmon_fan1_input_show, NULL);
> +
> +static struct attribute *hwmon_attributes[] = {
> + &hwmon_fan1_input.attr,
> + NULL
> +};
> +
> +static const struct attribute_group hwmon_attr_group = {
> + .attrs = hwmon_attributes,
> +};
> +
> +static void hwmon_exit(void)
> +{
> + if (!hwmon_device)
> + return;
> +
> + sysfs_remove_group(&hwmon_device->kobj, &hwmon_attr_group);
> + hwmon_device_unregister(hwmon_device);
> + hwmon_device = NULL;
> +}
> +
> +static int hwmon_init(void)
> +{
> + int res;
> +
> + hwmon_device = hwmon_device_register(&lenovo_sl_laptop_pdev->dev);
> + if (!hwmon_device) {
> + pr_err("Failed to register hwmon device\n");
> + return -ENODEV;
> + }
> +
> + res = sysfs_create_group(&hwmon_device->kobj, &hwmon_attr_group);
> + if (res < 0) {
> + pr_err("Failed to create hwmon sysfs group\n");
> + hwmon_device_unregister(hwmon_device);
> + hwmon_device = NULL;
> + return -ENODEV;
> + }
> + return 0;
> +}
> +
> +/*************************************************************************
> + hotkeys
> + *************************************************************************/
> +
> +typedef int (*acpi_ec_query_func) (void *data);
> +extern int acpi_ec_add_query_handler(void *ec, u8 query_bit,
> + acpi_handle handle,
> + acpi_ec_query_func func,
> + void *data);
> +extern void acpi_ec_remove_query_handler(void *ec, u8 query_bit);
> +
> +struct key_entry {
> + char type;
> + u8 scancode;
> + int keycode;
> +};
> +
> +enum { KE_KEY, KE_END };
> +
> +static struct input_dev *hkey_inputdev;
> +
> +static struct key_entry hkey_keymap[] = {
> + {KE_KEY, 0x0B, KEY_COFFEE },
> + {KE_KEY, 0x0C, KEY_BATTERY },
> + {KE_KEY, 0x0D, KEY_SLEEP },
> + {KE_KEY, 0x0E, KEY_WLAN },
> + {KE_KEY, 0x10, KEY_SWITCHVIDEOMODE },
> + {KE_KEY, 0x11, KEY_PROG1 },
> + {KE_KEY, 0x12, KEY_PROG2 },
> + {KE_KEY, 0x15, KEY_SUSPEND },
> + {KE_KEY, 0x69, KEY_VOLUMEUP },
> + {KE_KEY, 0x6A, KEY_VOLUMEDOWN },
> + {KE_KEY, 0x6B, KEY_MUTE },
> + {KE_KEY, 0x6C, KEY_BRIGHTNESSDOWN },
> + {KE_KEY, 0x6D, KEY_BRIGHTNESSUP },
> + {KE_KEY, 0x71, KEY_ZOOM },
> + {KE_KEY, 0x80, KEY_VENDOR },
> + {KE_END, 0},
> +};
> +
> +static int hkey_action(void *data)
> +{
> + int keycode;
> + struct key_entry *this_key = data;
> +
> + if (!data)
> + return -EINVAL;
> + keycode = this_key->keycode;
> +
> + if (keycode != KEY_RESERVED) {
> + input_report_key(hkey_inputdev, keycode, 1);
> + input_sync(hkey_inputdev);
> + input_report_key(hkey_inputdev, keycode, 0);
> + input_sync(hkey_inputdev);
> + }
> +
> + return 0;
> +}
> +
> +static int hkey_add(struct acpi_device *device)
> +{
> + int result;
> + struct key_entry *key;
> +
> + for (key = hkey_keymap; key->type != KE_END; key++) {
> + result = acpi_ec_add_query_handler(
> + acpi_driver_data(device->parent),
> + key->scancode, NULL,
> + hkey_action, key);
> + if (result) {
> + pr_err("Failed to register hotkey notification.\n");
> + return -ENODEV;
> + }
> + }
> + return 0;
> +}
> +
> +static int hkey_remove(struct acpi_device *device, int type)
> +{
> + struct key_entry *key;
> +
> + for (key = hkey_keymap; key->type != KE_END; key++) {
> + acpi_ec_remove_query_handler(
> + acpi_driver_data(device->parent),
> + key->scancode);
> + }
> + return 0;
> +}
> +
> +static const struct acpi_device_id hkey_ids[] = {
> + {"LEN0014", 0},
> + {"", 0},
> +};
> +
> +static struct acpi_driver hkey_driver = {
> + .name = "lenovo-sl-laptop-hotkey",
> + .class = "lenovo",
> + .ids = hkey_ids,
> + .owner = THIS_MODULE,
> + .ops = {
> + .add = hkey_add,
> + .remove = hkey_remove,
> + },
> +};
> +
> +static void hkey_inputdev_exit(void)
> +{
> + if (hkey_inputdev)
> + input_unregister_device(hkey_inputdev);
> +}
> +
> +static int hkey_inputdev_init(void)
> +{
> + int result;
> + struct key_entry *key;
> +
> + hkey_inputdev = input_allocate_device();
> + if (!hkey_inputdev) {
> + pr_err("Failed to allocate hotkey input device\n");
> + return -ENODEV;
> + }
> + hkey_inputdev->name = "Lenovo ThinkPad SL Series extra buttons";
> + hkey_inputdev->phys = LENOVO_SL_MODULE_NAME "/input0";
> + hkey_inputdev->uniq = LENOVO_SL_MODULE_NAME;
> + hkey_inputdev->id.bustype = BUS_HOST;
> + hkey_inputdev->id.vendor = PCI_VENDOR_ID_LENOVO;
> + hkey_inputdev->dev.parent = &lenovo_sl_laptop_pdev->dev;
> + set_bit(EV_KEY, hkey_inputdev->evbit);
> +
> + for (key = hkey_keymap; key->type != KE_END; key++)
> + set_bit(key->keycode, hkey_inputdev->keybit);
> +
> + result = input_register_device(hkey_inputdev);
> + if (result) {
> + pr_err("Failed to register hotkey input device\n");
> + input_free_device(hkey_inputdev);
> + hkey_inputdev = NULL;
> + return -ENODEV;
> + }
> + return 0;
> +}
> +
> +static void hkey_init(void)
> +{
> + int result;
> +
> + result = hkey_inputdev_init();
> + if (result) {
> + pr_err("Failed to register input device for hotkeys\n");
> + return;
> + }
> + result = acpi_bus_register_driver(&hkey_driver);
> + if (result)
> + pr_err("Failed to register hotkey driver\n");
> + return;
> +}
> +
> +static void hkey_exit(void)
> +{
> + hkey_inputdev_exit();
> + acpi_bus_unregister_driver(&hkey_driver);
> +}
> +
> +/*************************************************************************
> + init/exit
> + *************************************************************************/
> +
> +static int __init lenovo_sl_laptop_init(void)
> +{
> + int ret;
> + acpi_status status;
> +
> + if (acpi_disabled)
> + return -ENODEV;
> +
> + lenovo_sl_laptop_hkey_handle = lenovo_sl_laptop_ec0_handle = NULL;
> + status = acpi_get_handle(NULL, ACPI_HKEY_PATH,
> + &lenovo_sl_laptop_hkey_handle);
> + if (ACPI_FAILURE(status)) {
> + pr_err("Failed to get ACPI handle for %s\n", ACPI_HKEY_PATH);
> + return -ENODEV;
> + }
> + status = acpi_get_handle(NULL, ACPI_EC0_PATH,
> + &lenovo_sl_laptop_ec0_handle);
> + if (ACPI_FAILURE(status)) {
> + pr_err("Failed to get ACPI handle for %s\n", ACPI_EC0_PATH);
> + return -ENODEV;
> + }
> +
> + lenovo_sl_laptop_pdev = platform_device_register_simple(
> + LENOVO_SL_MODULE_NAME,
> + -1, NULL, 0);
> + if (IS_ERR(lenovo_sl_laptop_pdev)) {
> + ret = PTR_ERR(lenovo_sl_laptop_pdev);
> + lenovo_sl_laptop_pdev = NULL;
> + pr_err("Failed to register platform device\n");
> + return ret;
> + }
> +
> + radio_init(RADIO_BLUETOOTH);
> + radio_init(RADIO_WWAN);
> + radio_init(RADIO_UWB);
> +
> + hkey_init();
> + led_init();
> + hwmon_init();
> +
> + return 0;
> +}
> +
> +static void __exit lenovo_sl_laptop_exit(void)
> +{
> + hwmon_exit();
> + led_exit();
> + hkey_exit();
> +
> + radio_exit(RADIO_UWB);
> + radio_exit(RADIO_WWAN);
> + radio_exit(RADIO_BLUETOOTH);
> +
> + if (lenovo_sl_laptop_pdev)
> + platform_device_unregister(lenovo_sl_laptop_pdev);
> +}
> +
> +MODULE_DEVICE_TABLE(acpi, hkey_ids);
> +
> +module_init(lenovo_sl_laptop_init);
> +module_exit(lenovo_sl_laptop_exit);
>
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@...r.kernel.org
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/
Powered by blists - more mailing lists