Merge branch 'core-urgent-for-linus' of git://git.kernel.org/pub/scm/linux/kernel...
[cascardo/linux.git] / drivers / platform / x86 / asus-wireless.c
index 9ec721e..9f31bc1 100644 (file)
 #include <linux/acpi.h>
 #include <linux/input.h>
 #include <linux/pci_ids.h>
+#include <linux/leds.h>
+
+#define ASUS_WIRELESS_LED_STATUS 0x2
+#define ASUS_WIRELESS_LED_OFF 0x4
+#define ASUS_WIRELESS_LED_ON 0x5
 
 struct asus_wireless_data {
        struct input_dev *idev;
+       struct acpi_device *adev;
+       struct workqueue_struct *wq;
+       struct work_struct led_work;
+       struct led_classdev led;
+       int led_state;
 };
 
+static u64 asus_wireless_method(acpi_handle handle, const char *method,
+                               int param)
+{
+       struct acpi_object_list p;
+       union acpi_object obj;
+       acpi_status s;
+       u64 ret;
+
+       acpi_handle_debug(handle, "Evaluating method %s, parameter %#x\n",
+                         method, param);
+       obj.type = ACPI_TYPE_INTEGER;
+       obj.integer.value = param;
+       p.count = 1;
+       p.pointer = &obj;
+
+       s = acpi_evaluate_integer(handle, (acpi_string) method, &p, &ret);
+       if (ACPI_FAILURE(s))
+               acpi_handle_err(handle,
+                               "Failed to eval method %s, param %#x (%d)\n",
+                               method, param, s);
+       acpi_handle_debug(handle, "%s returned %#x\n", method, (uint) ret);
+       return ret;
+}
+
+static enum led_brightness led_state_get(struct led_classdev *led)
+{
+       struct asus_wireless_data *data;
+       int s;
+
+       data = container_of(led, struct asus_wireless_data, led);
+       s = asus_wireless_method(acpi_device_handle(data->adev), "HSWC",
+                                ASUS_WIRELESS_LED_STATUS);
+       if (s == ASUS_WIRELESS_LED_ON)
+               return LED_FULL;
+       return LED_OFF;
+}
+
+static void led_state_update(struct work_struct *work)
+{
+       struct asus_wireless_data *data;
+
+       data = container_of(work, struct asus_wireless_data, led_work);
+       asus_wireless_method(acpi_device_handle(data->adev), "HSWC",
+                            data->led_state);
+}
+
+static void led_state_set(struct led_classdev *led,
+                                 enum led_brightness value)
+{
+       struct asus_wireless_data *data;
+
+       data = container_of(led, struct asus_wireless_data, led);
+       data->led_state = value == LED_OFF ? ASUS_WIRELESS_LED_OFF :
+                                            ASUS_WIRELESS_LED_ON;
+       queue_work(data->wq, &data->led_work);
+}
+
 static void asus_wireless_notify(struct acpi_device *adev, u32 event)
 {
        struct asus_wireless_data *data = acpi_driver_data(adev);
@@ -37,6 +104,7 @@ static void asus_wireless_notify(struct acpi_device *adev, u32 event)
 static int asus_wireless_add(struct acpi_device *adev)
 {
        struct asus_wireless_data *data;
+       int err;
 
        data = devm_kzalloc(&adev->dev, sizeof(*data), GFP_KERNEL);
        if (!data)
@@ -52,11 +120,32 @@ static int asus_wireless_add(struct acpi_device *adev)
        data->idev->id.vendor = PCI_VENDOR_ID_ASUSTEK;
        set_bit(EV_KEY, data->idev->evbit);
        set_bit(KEY_RFKILL, data->idev->keybit);
-       return input_register_device(data->idev);
+       err = input_register_device(data->idev);
+       if (err)
+               return err;
+
+       data->adev = adev;
+       data->wq = create_singlethread_workqueue("asus_wireless_workqueue");
+       if (!data->wq)
+               return -ENOMEM;
+       INIT_WORK(&data->led_work, led_state_update);
+       data->led.name = "asus-wireless::airplane";
+       data->led.brightness_set = led_state_set;
+       data->led.brightness_get = led_state_get;
+       data->led.flags = LED_CORE_SUSPENDRESUME;
+       data->led.max_brightness = 1;
+       err = devm_led_classdev_register(&adev->dev, &data->led);
+       if (err)
+               destroy_workqueue(data->wq);
+       return err;
 }
 
 static int asus_wireless_remove(struct acpi_device *adev)
 {
+       struct asus_wireless_data *data = acpi_driver_data(adev);
+
+       if (data->wq)
+               destroy_workqueue(data->wq);
        return 0;
 }