4   Copyright (C) 2007,2008 Jonathan Woithe <jwoithe@physics.adelaide.edu.au>
 
   5   Copyright (C) 2008 Peter Gruber <nokos@gmx.net>
 
   7     Copyright (C) 2003 Shane Spencer <shane@bogomip.com>
 
   8     Adrian Yee <brewt-fujitsu@brewt.org>
 
  10   Templated from msi-laptop.c and thinkpad_acpi.c which is copyright
 
  11   by its respective authors.
 
  13   This program is free software; you can redistribute it and/or modify
 
  14   it under the terms of the GNU General Public License as published by
 
  15   the Free Software Foundation; either version 2 of the License, or
 
  16   (at your option) any later version.
 
  18   This program is distributed in the hope that it will be useful, but
 
  19   WITHOUT ANY WARRANTY; without even the implied warranty of
 
  20   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 
  21   General Public License for more details.
 
  23   You should have received a copy of the GNU General Public License
 
  24   along with this program; if not, write to the Free Software
 
  25   Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
 
  30  * fujitsu-laptop.c - Fujitsu laptop support, providing access to additional
 
  31  * features made available on a range of Fujitsu laptops including the
 
  32  * P2xxx/P5xxx/S6xxx/S7xxx series.
 
  34  * This driver exports a few files in /sys/devices/platform/fujitsu-laptop/;
 
  35  * others may be added at a later date.
 
  37  *   lcd_level - Screen brightness: contains a single integer in the
 
  40  * In addition to these platform device attributes the driver
 
  41  * registers itself in the Linux backlight control subsystem and is
 
  42  * available to userspace under /sys/class/backlight/fujitsu-laptop/.
 
  44  * Hotkeys present on certain Fujitsu laptops (eg: the S6xxx series) are
 
  45  * also supported by this driver.
 
  47  * This driver has been tested on a Fujitsu Lifebook S6410 and S7020.  It
 
  48  * should work on most P-series and S-series Lifebooks, but YMMV.
 
  50  * The module parameter use_alt_lcd_levels switches between different ACPI
 
  51  * brightness controls which are used by different Fujitsu laptops.  In most
 
  52  * cases the correct method is automatically detected. "use_alt_lcd_levels=1"
 
  53  * is applicable for a Fujitsu Lifebook S6410 if autodetection fails.
 
  57 #include <linux/module.h>
 
  58 #include <linux/kernel.h>
 
  59 #include <linux/init.h>
 
  60 #include <linux/acpi.h>
 
  61 #include <linux/dmi.h>
 
  62 #include <linux/backlight.h>
 
  63 #include <linux/input.h>
 
  64 #include <linux/kfifo.h>
 
  65 #include <linux/video_output.h>
 
  66 #include <linux/platform_device.h>
 
  68 #define FUJITSU_DRIVER_VERSION "0.4.2"
 
  70 #define FUJITSU_LCD_N_LEVELS 8
 
  72 #define ACPI_FUJITSU_CLASS              "fujitsu"
 
  73 #define ACPI_FUJITSU_HID                "FUJ02B1"
 
  74 #define ACPI_FUJITSU_DRIVER_NAME        "Fujitsu laptop FUJ02B1 ACPI brightness driver"
 
  75 #define ACPI_FUJITSU_DEVICE_NAME        "Fujitsu FUJ02B1"
 
  76 #define ACPI_FUJITSU_HOTKEY_HID         "FUJ02E3"
 
  77 #define ACPI_FUJITSU_HOTKEY_DRIVER_NAME "Fujitsu laptop FUJ02E3 ACPI hotkeys driver"
 
  78 #define ACPI_FUJITSU_HOTKEY_DEVICE_NAME "Fujitsu FUJ02E3"
 
  80 #define ACPI_FUJITSU_NOTIFY_CODE1     0x80
 
  82 #define ACPI_VIDEO_NOTIFY_INC_BRIGHTNESS     0x86
 
  83 #define ACPI_VIDEO_NOTIFY_DEC_BRIGHTNESS     0x87
 
  86 #define LOCK_KEY        0x410   /* codes for the keys in the GIRB register */
 
  87 #define DISPLAY_KEY     0x411   /* keys are mapped to KEY_SCREENLOCK (the key with the key symbol) */
 
  88 #define ENERGY_KEY      0x412   /* KEY_MEDIA (the key with the laptop symbol, KEY_EMAIL (E key)) */
 
  89 #define REST_KEY        0x413   /* KEY_SUSPEND (R key) */
 
  91 #define MAX_HOTKEY_RINGBUFFER_SIZE 100
 
  92 #define RINGBUFFERSIZE 40
 
  95 #define FUJLAPTOP_LOG      ACPI_FUJITSU_HID ": "
 
  96 #define FUJLAPTOP_ERR      KERN_ERR FUJLAPTOP_LOG
 
  97 #define FUJLAPTOP_NOTICE   KERN_NOTICE FUJLAPTOP_LOG
 
  98 #define FUJLAPTOP_INFO     KERN_INFO FUJLAPTOP_LOG
 
  99 #define FUJLAPTOP_DEBUG    KERN_DEBUG FUJLAPTOP_LOG
 
 101 #define FUJLAPTOP_DBG_ALL         0xffff
 
 102 #define FUJLAPTOP_DBG_ERROR       0x0001
 
 103 #define FUJLAPTOP_DBG_WARN        0x0002
 
 104 #define FUJLAPTOP_DBG_INFO        0x0004
 
 105 #define FUJLAPTOP_DBG_TRACE       0x0008
 
 107 #define dbg_printk(a_dbg_level, format, arg...) \
 
 108         do { if (dbg_level & a_dbg_level) \
 
 109                 printk(FUJLAPTOP_DEBUG "%s: " format, __func__ , ## arg); \
 
 111 #ifdef CONFIG_FUJITSU_LAPTOP_DEBUG
 
 112 #define vdbg_printk(a_dbg_level, format, arg...) \
 
 113         dbg_printk(a_dbg_level, format, ## arg)
 
 115 #define vdbg_printk(a_dbg_level, format, arg...)
 
 118 /* Device controlling the backlight and associated keys */
 
 120         acpi_handle acpi_handle;
 
 121         struct acpi_device *dev;
 
 122         struct input_dev *input;
 
 124         struct backlight_device *bl_device;
 
 125         struct platform_device *pf_device;
 
 127         unsigned int max_brightness;
 
 128         unsigned int brightness_changed;
 
 129         unsigned int brightness_level;
 
 132 static struct fujitsu_t *fujitsu;
 
 133 static int use_alt_lcd_levels = -1;
 
 134 static int disable_brightness_keys = -1;
 
 135 static int disable_brightness_adjust = -1;
 
 137 /* Device used to access other hotkeys on the laptop */
 
 138 struct fujitsu_hotkey_t {
 
 139         acpi_handle acpi_handle;
 
 140         struct acpi_device *dev;
 
 141         struct input_dev *input;
 
 143         struct platform_device *pf_device;
 
 145         spinlock_t fifo_lock;
 
 147         unsigned int irb;       /* info about the pressed buttons */
 
 150 static struct fujitsu_hotkey_t *fujitsu_hotkey;
 
 152 static void acpi_fujitsu_hotkey_notify(acpi_handle handle, u32 event,
 
 155 #ifdef CONFIG_FUJITSU_LAPTOP_DEBUG
 
 156 static u32 dbg_level = 0x03;
 
 159 static void acpi_fujitsu_notify(acpi_handle handle, u32 event, void *data);
 
 161 /* Hardware access for LCD brightness control */
 
 163 static int set_lcd_level(int level)
 
 165         acpi_status status = AE_OK;
 
 166         union acpi_object arg0 = { ACPI_TYPE_INTEGER };
 
 167         struct acpi_object_list arg_list = { 1, &arg0 };
 
 168         acpi_handle handle = NULL;
 
 170         vdbg_printk(FUJLAPTOP_DBG_TRACE, "set lcd level via SBLL [%d]\n",
 
 173         if (level < 0 || level >= fujitsu->max_brightness)
 
 179         status = acpi_get_handle(fujitsu->acpi_handle, "SBLL", &handle);
 
 180         if (ACPI_FAILURE(status)) {
 
 181                 vdbg_printk(FUJLAPTOP_DBG_ERROR, "SBLL not present\n");
 
 185         arg0.integer.value = level;
 
 187         status = acpi_evaluate_object(handle, NULL, &arg_list, NULL);
 
 188         if (ACPI_FAILURE(status))
 
 194 static int set_lcd_level_alt(int level)
 
 196         acpi_status status = AE_OK;
 
 197         union acpi_object arg0 = { ACPI_TYPE_INTEGER };
 
 198         struct acpi_object_list arg_list = { 1, &arg0 };
 
 199         acpi_handle handle = NULL;
 
 201         vdbg_printk(FUJLAPTOP_DBG_TRACE, "set lcd level via SBL2 [%d]\n",
 
 204         if (level < 0 || level >= fujitsu->max_brightness)
 
 210         status = acpi_get_handle(fujitsu->acpi_handle, "SBL2", &handle);
 
 211         if (ACPI_FAILURE(status)) {
 
 212                 vdbg_printk(FUJLAPTOP_DBG_ERROR, "SBL2 not present\n");
 
 216         arg0.integer.value = level;
 
 218         status = acpi_evaluate_object(handle, NULL, &arg_list, NULL);
 
 219         if (ACPI_FAILURE(status))
 
 225 static int get_lcd_level(void)
 
 227         unsigned long state = 0;
 
 228         acpi_status status = AE_OK;
 
 230         vdbg_printk(FUJLAPTOP_DBG_TRACE, "get lcd level via GBLL\n");
 
 233             acpi_evaluate_integer(fujitsu->acpi_handle, "GBLL", NULL, &state);
 
 237         fujitsu->brightness_level = state & 0x0fffffff;
 
 239         if (state & 0x80000000)
 
 240                 fujitsu->brightness_changed = 1;
 
 242                 fujitsu->brightness_changed = 0;
 
 244         return fujitsu->brightness_level;
 
 247 static int get_max_brightness(void)
 
 249         unsigned long state = 0;
 
 250         acpi_status status = AE_OK;
 
 252         vdbg_printk(FUJLAPTOP_DBG_TRACE, "get max lcd level via RBLL\n");
 
 255             acpi_evaluate_integer(fujitsu->acpi_handle, "RBLL", NULL, &state);
 
 259         fujitsu->max_brightness = state;
 
 261         return fujitsu->max_brightness;
 
 264 static int get_lcd_level_alt(void)
 
 266         unsigned long state = 0;
 
 267         acpi_status status = AE_OK;
 
 269         vdbg_printk(FUJLAPTOP_DBG_TRACE, "get lcd level via GBLS\n");
 
 272             acpi_evaluate_integer(fujitsu->acpi_handle, "GBLS", NULL, &state);
 
 276         fujitsu->brightness_level = state & 0x0fffffff;
 
 278         if (state & 0x80000000)
 
 279                 fujitsu->brightness_changed = 1;
 
 281                 fujitsu->brightness_changed = 0;
 
 283         return fujitsu->brightness_level;
 
 286 /* Backlight device stuff */
 
 288 static int bl_get_brightness(struct backlight_device *b)
 
 290         if (use_alt_lcd_levels)
 
 291                 return get_lcd_level_alt();
 
 293                 return get_lcd_level();
 
 296 static int bl_update_status(struct backlight_device *b)
 
 298         if (use_alt_lcd_levels)
 
 299                 return set_lcd_level_alt(b->props.brightness);
 
 301                 return set_lcd_level(b->props.brightness);
 
 304 static struct backlight_ops fujitsubl_ops = {
 
 305         .get_brightness = bl_get_brightness,
 
 306         .update_status = bl_update_status,
 
 309 /* Platform LCD brightness device */
 
 312 show_max_brightness(struct device *dev,
 
 313                     struct device_attribute *attr, char *buf)
 
 318         ret = get_max_brightness();
 
 322         return sprintf(buf, "%i\n", ret);
 
 326 show_brightness_changed(struct device *dev,
 
 327                         struct device_attribute *attr, char *buf)
 
 332         ret = fujitsu->brightness_changed;
 
 336         return sprintf(buf, "%i\n", ret);
 
 339 static ssize_t show_lcd_level(struct device *dev,
 
 340                               struct device_attribute *attr, char *buf)
 
 345         if (use_alt_lcd_levels)
 
 346                 ret = get_lcd_level_alt();
 
 348                 ret = get_lcd_level();
 
 352         return sprintf(buf, "%i\n", ret);
 
 355 static ssize_t store_lcd_level(struct device *dev,
 
 356                                struct device_attribute *attr, const char *buf,
 
 362         if (sscanf(buf, "%i", &level) != 1
 
 363             || (level < 0 || level >= fujitsu->max_brightness))
 
 366         if (use_alt_lcd_levels)
 
 367                 ret = set_lcd_level_alt(level);
 
 369                 ret = set_lcd_level(level);
 
 373         if (use_alt_lcd_levels)
 
 374                 ret = get_lcd_level_alt();
 
 376                 ret = get_lcd_level();
 
 383 /* Hardware access for hotkey device */
 
 385 static int get_irb(void)
 
 387         unsigned long state = 0;
 
 388         acpi_status status = AE_OK;
 
 390         vdbg_printk(FUJLAPTOP_DBG_TRACE, "Get irb\n");
 
 393             acpi_evaluate_integer(fujitsu_hotkey->acpi_handle, "GIRB", NULL,
 
 398         fujitsu_hotkey->irb = state;
 
 400         return fujitsu_hotkey->irb;
 
 404 ignore_store(struct device *dev,
 
 405              struct device_attribute *attr, const char *buf, size_t count)
 
 410 static DEVICE_ATTR(max_brightness, 0444, show_max_brightness, ignore_store);
 
 411 static DEVICE_ATTR(brightness_changed, 0444, show_brightness_changed,
 
 413 static DEVICE_ATTR(lcd_level, 0644, show_lcd_level, store_lcd_level);
 
 415 static struct attribute *fujitsupf_attributes[] = {
 
 416         &dev_attr_brightness_changed.attr,
 
 417         &dev_attr_max_brightness.attr,
 
 418         &dev_attr_lcd_level.attr,
 
 422 static struct attribute_group fujitsupf_attribute_group = {
 
 423         .attrs = fujitsupf_attributes
 
 426 static struct platform_driver fujitsupf_driver = {
 
 428                    .name = "fujitsu-laptop",
 
 429                    .owner = THIS_MODULE,
 
 433 static int dmi_check_cb_s6410(const struct dmi_system_id *id)
 
 437         printk(KERN_INFO "fujitsu-laptop: Identified laptop model '%s'.\n",
 
 439         have_blnf = ACPI_SUCCESS
 
 440             (acpi_get_handle(NULL, "\\_SB.PCI0.GFX0.LCD.BLNF", &handle));
 
 441         if (use_alt_lcd_levels == -1) {
 
 442                 vdbg_printk(FUJLAPTOP_DBG_TRACE, "auto-detecting usealt\n");
 
 443                 use_alt_lcd_levels = 1;
 
 445         if (disable_brightness_keys == -1) {
 
 446                 vdbg_printk(FUJLAPTOP_DBG_TRACE,
 
 447                             "auto-detecting disable_keys\n");
 
 448                 disable_brightness_keys = have_blnf ? 1 : 0;
 
 450         if (disable_brightness_adjust == -1) {
 
 451                 vdbg_printk(FUJLAPTOP_DBG_TRACE,
 
 452                             "auto-detecting disable_adjust\n");
 
 453                 disable_brightness_adjust = have_blnf ? 0 : 1;
 
 458 static struct dmi_system_id __initdata fujitsu_dmi_table[] = {
 
 460          .ident = "Fujitsu Siemens",
 
 462                      DMI_MATCH(DMI_SYS_VENDOR, "FUJITSU SIEMENS"),
 
 463                      DMI_MATCH(DMI_PRODUCT_NAME, "LIFEBOOK S6410"),
 
 465          .callback = dmi_check_cb_s6410},
 
 467          .ident = "FUJITSU LifeBook P8010",
 
 469                      DMI_MATCH(DMI_SYS_VENDOR, "FUJITSU"),
 
 470                      DMI_MATCH(DMI_PRODUCT_NAME, "LifeBook P8010"),
 
 472          .callback = dmi_check_cb_s6410},
 
 476 /* ACPI device for LCD brightness control */
 
 478 static int acpi_fujitsu_add(struct acpi_device *device)
 
 484         struct input_dev *input;
 
 490         fujitsu->acpi_handle = device->handle;
 
 491         sprintf(acpi_device_name(device), "%s", ACPI_FUJITSU_DEVICE_NAME);
 
 492         sprintf(acpi_device_class(device), "%s", ACPI_FUJITSU_CLASS);
 
 493         acpi_driver_data(device) = fujitsu;
 
 495         status = acpi_install_notify_handler(device->handle,
 
 497                                              acpi_fujitsu_notify, fujitsu);
 
 499         if (ACPI_FAILURE(status)) {
 
 500                 printk(KERN_ERR "Error installing notify handler\n");
 
 505         fujitsu->input = input = input_allocate_device();
 
 508                 goto err_uninstall_notify;
 
 511         snprintf(fujitsu->phys, sizeof(fujitsu->phys),
 
 512                  "%s/video/input0", acpi_device_hid(device));
 
 514         input->name = acpi_device_name(device);
 
 515         input->phys = fujitsu->phys;
 
 516         input->id.bustype = BUS_HOST;
 
 517         input->id.product = 0x06;
 
 518         input->dev.parent = &device->dev;
 
 519         input->evbit[0] = BIT(EV_KEY);
 
 520         set_bit(KEY_BRIGHTNESSUP, input->keybit);
 
 521         set_bit(KEY_BRIGHTNESSDOWN, input->keybit);
 
 522         set_bit(KEY_UNKNOWN, input->keybit);
 
 524         error = input_register_device(input);
 
 526                 goto err_free_input_dev;
 
 528         result = acpi_bus_get_power(fujitsu->acpi_handle, &state);
 
 530                 printk(KERN_ERR "Error reading power state\n");
 
 534         printk(KERN_INFO PREFIX "%s [%s] (%s)\n",
 
 535                acpi_device_name(device), acpi_device_bid(device),
 
 536                !device->power.state ? "on" : "off");
 
 538         fujitsu->dev = device;
 
 541             (acpi_get_handle(device->handle, METHOD_NAME__INI, &handle))) {
 
 542                 vdbg_printk(FUJLAPTOP_DBG_INFO, "Invoking _INI\n");
 
 544                     (acpi_evaluate_object
 
 545                      (device->handle, METHOD_NAME__INI, NULL, NULL)))
 
 546                         printk(KERN_ERR "_INI Method failed\n");
 
 549         /* do config (detect defaults) */
 
 550         dmi_check_system(fujitsu_dmi_table);
 
 551         use_alt_lcd_levels = use_alt_lcd_levels == 1 ? 1 : 0;
 
 552         disable_brightness_keys = disable_brightness_keys == 1 ? 1 : 0;
 
 553         disable_brightness_adjust = disable_brightness_adjust == 1 ? 1 : 0;
 
 554         vdbg_printk(FUJLAPTOP_DBG_INFO,
 
 555                     "config: [alt interface: %d], [key disable: %d], [adjust disable: %d]\n",
 
 556                     use_alt_lcd_levels, disable_brightness_keys,
 
 557                     disable_brightness_adjust);
 
 559         if (get_max_brightness() <= 0)
 
 560                 fujitsu->max_brightness = FUJITSU_LCD_N_LEVELS;
 
 561         if (use_alt_lcd_levels)
 
 570         input_free_device(input);
 
 571 err_uninstall_notify:
 
 572         acpi_remove_notify_handler(device->handle, ACPI_DEVICE_NOTIFY,
 
 573                                    acpi_fujitsu_notify);
 
 579 static int acpi_fujitsu_remove(struct acpi_device *device, int type)
 
 582         struct fujitsu_t *fujitsu = NULL;
 
 584         if (!device || !acpi_driver_data(device))
 
 587         fujitsu = acpi_driver_data(device);
 
 589         status = acpi_remove_notify_handler(fujitsu->acpi_handle,
 
 591                                             acpi_fujitsu_notify);
 
 593         if (!device || !acpi_driver_data(device))
 
 596         fujitsu->acpi_handle = NULL;
 
 601 /* Brightness notify */
 
 603 static void acpi_fujitsu_notify(acpi_handle handle, u32 event, void *data)
 
 605         struct input_dev *input;
 
 609         input = fujitsu->input;
 
 612         case ACPI_FUJITSU_NOTIFY_CODE1:
 
 614                 oldb = fujitsu->brightness_level;
 
 615                 get_lcd_level();  /* the alt version always yields changed */
 
 616                 newb = fujitsu->brightness_level;
 
 618                 vdbg_printk(FUJLAPTOP_DBG_TRACE,
 
 619                             "brightness button event [%i -> %i (%i)]\n",
 
 620                             oldb, newb, fujitsu->brightness_changed);
 
 622                 if (oldb == newb && fujitsu->brightness_changed) {
 
 624                         if (disable_brightness_keys != 1) {
 
 626                                         acpi_bus_generate_proc_event(fujitsu->
 
 628                                                 ACPI_VIDEO_NOTIFY_DEC_BRIGHTNESS,
 
 630                                         keycode = KEY_BRIGHTNESSDOWN;
 
 632                                            (fujitsu->max_brightness) - 1) {
 
 633                                         acpi_bus_generate_proc_event(fujitsu->
 
 635                                                 ACPI_VIDEO_NOTIFY_INC_BRIGHTNESS,
 
 637                                         keycode = KEY_BRIGHTNESSUP;
 
 640                 } else if (oldb < newb) {
 
 641                         if (disable_brightness_adjust != 1) {
 
 642                                 if (use_alt_lcd_levels)
 
 643                                         set_lcd_level_alt(newb);
 
 647                         if (disable_brightness_keys != 1) {
 
 648                                 acpi_bus_generate_proc_event(fujitsu->dev,
 
 649                                         ACPI_VIDEO_NOTIFY_INC_BRIGHTNESS,
 
 651                                 keycode = KEY_BRIGHTNESSUP;
 
 653                 } else if (oldb > newb) {
 
 654                         if (disable_brightness_adjust != 1) {
 
 655                                 if (use_alt_lcd_levels)
 
 656                                         set_lcd_level_alt(newb);
 
 660                         if (disable_brightness_keys != 1) {
 
 661                                 acpi_bus_generate_proc_event(fujitsu->dev,
 
 662                                         ACPI_VIDEO_NOTIFY_DEC_BRIGHTNESS,
 
 664                                 keycode = KEY_BRIGHTNESSDOWN;
 
 667                         keycode = KEY_UNKNOWN;
 
 671                 keycode = KEY_UNKNOWN;
 
 672                 vdbg_printk(FUJLAPTOP_DBG_WARN,
 
 673                             "unsupported event [0x%x]\n", event);
 
 678                 input_report_key(input, keycode, 1);
 
 680                 input_report_key(input, keycode, 0);
 
 687 /* ACPI device for hotkey handling */
 
 689 static int acpi_fujitsu_hotkey_add(struct acpi_device *device)
 
 695         struct input_dev *input;
 
 702         fujitsu_hotkey->acpi_handle = device->handle;
 
 703         sprintf(acpi_device_name(device), "%s",
 
 704                 ACPI_FUJITSU_HOTKEY_DEVICE_NAME);
 
 705         sprintf(acpi_device_class(device), "%s", ACPI_FUJITSU_CLASS);
 
 706         acpi_driver_data(device) = fujitsu_hotkey;
 
 708         status = acpi_install_notify_handler(device->handle,
 
 710                                              acpi_fujitsu_hotkey_notify,
 
 713         if (ACPI_FAILURE(status)) {
 
 714                 printk(KERN_ERR "Error installing notify handler\n");
 
 720         spin_lock_init(&fujitsu_hotkey->fifo_lock);
 
 721         fujitsu_hotkey->fifo =
 
 722             kfifo_alloc(RINGBUFFERSIZE * sizeof(int), GFP_KERNEL,
 
 723                         &fujitsu_hotkey->fifo_lock);
 
 724         if (IS_ERR(fujitsu_hotkey->fifo)) {
 
 725                 printk(KERN_ERR "kfifo_alloc failed\n");
 
 726                 error = PTR_ERR(fujitsu_hotkey->fifo);
 
 730         fujitsu_hotkey->input = input = input_allocate_device();
 
 733                 goto err_uninstall_notify;
 
 736         snprintf(fujitsu_hotkey->phys, sizeof(fujitsu_hotkey->phys),
 
 737                  "%s/video/input0", acpi_device_hid(device));
 
 739         input->name = acpi_device_name(device);
 
 740         input->phys = fujitsu_hotkey->phys;
 
 741         input->id.bustype = BUS_HOST;
 
 742         input->id.product = 0x06;
 
 743         input->dev.parent = &device->dev;
 
 744         input->evbit[0] = BIT(EV_KEY);
 
 745         set_bit(KEY_SCREENLOCK, input->keybit);
 
 746         set_bit(KEY_MEDIA, input->keybit);
 
 747         set_bit(KEY_EMAIL, input->keybit);
 
 748         set_bit(KEY_SUSPEND, input->keybit);
 
 749         set_bit(KEY_UNKNOWN, input->keybit);
 
 751         error = input_register_device(input);
 
 753                 goto err_free_input_dev;
 
 755         result = acpi_bus_get_power(fujitsu_hotkey->acpi_handle, &state);
 
 757                 printk(KERN_ERR "Error reading power state\n");
 
 761         printk(KERN_INFO PREFIX "%s [%s] (%s)\n",
 
 762                acpi_device_name(device), acpi_device_bid(device),
 
 763                !device->power.state ? "on" : "off");
 
 765         fujitsu_hotkey->dev = device;
 
 768             (acpi_get_handle(device->handle, METHOD_NAME__INI, &handle))) {
 
 769                 vdbg_printk(FUJLAPTOP_DBG_INFO, "Invoking _INI\n");
 
 771                     (acpi_evaluate_object
 
 772                      (device->handle, METHOD_NAME__INI, NULL, NULL)))
 
 773                         printk(KERN_ERR "_INI Method failed\n");
 
 776         i = 0;                  /* Discard hotkey ringbuffer */
 
 777         while (get_irb() != 0 && (i++) < MAX_HOTKEY_RINGBUFFER_SIZE) ;
 
 778         vdbg_printk(FUJLAPTOP_DBG_INFO, "Discarded %i ringbuffer entries\n", i);
 
 784         input_free_device(input);
 
 785 err_uninstall_notify:
 
 786         acpi_remove_notify_handler(device->handle, ACPI_DEVICE_NOTIFY,
 
 787                                    acpi_fujitsu_hotkey_notify);
 
 788         kfifo_free(fujitsu_hotkey->fifo);
 
 794 static int acpi_fujitsu_hotkey_remove(struct acpi_device *device, int type)
 
 797         struct fujitsu_hotkey_t *fujitsu_hotkey = NULL;
 
 799         if (!device || !acpi_driver_data(device))
 
 802         fujitsu_hotkey = acpi_driver_data(device);
 
 804         status = acpi_remove_notify_handler(fujitsu_hotkey->acpi_handle,
 
 806                                             acpi_fujitsu_hotkey_notify);
 
 808         fujitsu_hotkey->acpi_handle = NULL;
 
 810         kfifo_free(fujitsu_hotkey->fifo);
 
 815 static void acpi_fujitsu_hotkey_notify(acpi_handle handle, u32 event,
 
 818         struct input_dev *input;
 
 819         int keycode, keycode_r;
 
 820         unsigned int irb = 1;
 
 823         input = fujitsu_hotkey->input;
 
 825         vdbg_printk(FUJLAPTOP_DBG_TRACE, "Hotkey event\n");
 
 828         case ACPI_FUJITSU_NOTIFY_CODE1:
 
 830                 while ((irb = get_irb()) != 0
 
 831                        && (i++) < MAX_HOTKEY_RINGBUFFER_SIZE) {
 
 832                         vdbg_printk(FUJLAPTOP_DBG_TRACE, "GIRB result [%x]\n",
 
 835                         switch (irb & 0x4ff) {
 
 837                                 keycode = KEY_SCREENLOCK;
 
 846                                 keycode = KEY_SUSPEND;
 
 852                                 vdbg_printk(FUJLAPTOP_DBG_WARN,
 
 853                                         "Unknown GIRB result [%x]\n", irb);
 
 858                                 vdbg_printk(FUJLAPTOP_DBG_TRACE,
 
 859                                         "Push keycode into ringbuffer [%d]\n",
 
 861                                 status = kfifo_put(fujitsu_hotkey->fifo,
 
 862                                                 (unsigned char *)&keycode,
 
 864                                 if (status != sizeof(keycode)) {
 
 865                                         vdbg_printk(FUJLAPTOP_DBG_WARN,
 
 866                                                 "Could not push keycode [0x%x]\n",
 
 869                                         input_report_key(input, keycode, 1);
 
 872                         } else if (keycode == 0) {
 
 875                                         (fujitsu_hotkey->fifo, (unsigned char *)
 
 878                                          (keycode_r))) == sizeof(keycode_r)) {
 
 879                                         input_report_key(input, keycode_r, 0);
 
 881                                         vdbg_printk(FUJLAPTOP_DBG_TRACE,
 
 882                                                     "Pop keycode from ringbuffer [%d]\n",
 
 890                 keycode = KEY_UNKNOWN;
 
 891                 vdbg_printk(FUJLAPTOP_DBG_WARN,
 
 892                             "Unsupported event [0x%x]\n", event);
 
 893                 input_report_key(input, keycode, 1);
 
 895                 input_report_key(input, keycode, 0);
 
 905 static const struct acpi_device_id fujitsu_device_ids[] = {
 
 906         {ACPI_FUJITSU_HID, 0},
 
 910 static struct acpi_driver acpi_fujitsu_driver = {
 
 911         .name = ACPI_FUJITSU_DRIVER_NAME,
 
 912         .class = ACPI_FUJITSU_CLASS,
 
 913         .ids = fujitsu_device_ids,
 
 915                 .add = acpi_fujitsu_add,
 
 916                 .remove = acpi_fujitsu_remove,
 
 920 static const struct acpi_device_id fujitsu_hotkey_device_ids[] = {
 
 921         {ACPI_FUJITSU_HOTKEY_HID, 0},
 
 925 static struct acpi_driver acpi_fujitsu_hotkey_driver = {
 
 926         .name = ACPI_FUJITSU_HOTKEY_DRIVER_NAME,
 
 927         .class = ACPI_FUJITSU_CLASS,
 
 928         .ids = fujitsu_hotkey_device_ids,
 
 930                 .add = acpi_fujitsu_hotkey_add,
 
 931                 .remove = acpi_fujitsu_hotkey_remove,
 
 935 static int __init fujitsu_init(void)
 
 937         int ret, result, max_brightness;
 
 942         fujitsu = kmalloc(sizeof(struct fujitsu_t), GFP_KERNEL);
 
 945         memset(fujitsu, 0, sizeof(struct fujitsu_t));
 
 947         result = acpi_bus_register_driver(&acpi_fujitsu_driver);
 
 953         /* Register platform stuff */
 
 955         fujitsu->pf_device = platform_device_alloc("fujitsu-laptop", -1);
 
 956         if (!fujitsu->pf_device) {
 
 958                 goto fail_platform_driver;
 
 961         ret = platform_device_add(fujitsu->pf_device);
 
 963                 goto fail_platform_device1;
 
 966             sysfs_create_group(&fujitsu->pf_device->dev.kobj,
 
 967                                &fujitsupf_attribute_group);
 
 969                 goto fail_platform_device2;
 
 971         /* Register backlight stuff */
 
 974             backlight_device_register("fujitsu-laptop", NULL, NULL,
 
 976         if (IS_ERR(fujitsu->bl_device))
 
 977                 return PTR_ERR(fujitsu->bl_device);
 
 979         max_brightness = fujitsu->max_brightness;
 
 981         fujitsu->bl_device->props.max_brightness = max_brightness - 1;
 
 982         fujitsu->bl_device->props.brightness = fujitsu->brightness_level;
 
 984         ret = platform_driver_register(&fujitsupf_driver);
 
 988         /* Register hotkey driver */
 
 990         fujitsu_hotkey = kmalloc(sizeof(struct fujitsu_hotkey_t), GFP_KERNEL);
 
 991         if (!fujitsu_hotkey) {
 
 995         memset(fujitsu_hotkey, 0, sizeof(struct fujitsu_hotkey_t));
 
 997         result = acpi_bus_register_driver(&acpi_fujitsu_hotkey_driver);
 
1003         printk(KERN_INFO "fujitsu-laptop: driver " FUJITSU_DRIVER_VERSION
 
1004                " successfully loaded.\n");
 
1010         kfree(fujitsu_hotkey);
 
1014         platform_driver_unregister(&fujitsupf_driver);
 
1018         backlight_device_unregister(fujitsu->bl_device);
 
1020 fail_platform_device2:
 
1022         platform_device_del(fujitsu->pf_device);
 
1024 fail_platform_device1:
 
1026         platform_device_put(fujitsu->pf_device);
 
1028 fail_platform_driver:
 
1030         acpi_bus_unregister_driver(&acpi_fujitsu_driver);
 
1039 static void __exit fujitsu_cleanup(void)
 
1041         sysfs_remove_group(&fujitsu->pf_device->dev.kobj,
 
1042                            &fujitsupf_attribute_group);
 
1043         platform_device_unregister(fujitsu->pf_device);
 
1044         platform_driver_unregister(&fujitsupf_driver);
 
1045         backlight_device_unregister(fujitsu->bl_device);
 
1047         acpi_bus_unregister_driver(&acpi_fujitsu_driver);
 
1051         acpi_bus_unregister_driver(&acpi_fujitsu_hotkey_driver);
 
1053         kfree(fujitsu_hotkey);
 
1055         printk(KERN_INFO "fujitsu-laptop: driver unloaded.\n");
 
1058 module_init(fujitsu_init);
 
1059 module_exit(fujitsu_cleanup);
 
1061 module_param(use_alt_lcd_levels, uint, 0644);
 
1062 MODULE_PARM_DESC(use_alt_lcd_levels,
 
1063                  "Use alternative interface for lcd_levels (needed for Lifebook s6410).");
 
1064 module_param(disable_brightness_keys, uint, 0644);
 
1065 MODULE_PARM_DESC(disable_brightness_keys,
 
1066                  "Disable brightness keys (eg. if they are already handled by the generic ACPI_VIDEO device).");
 
1067 module_param(disable_brightness_adjust, uint, 0644);
 
1068 MODULE_PARM_DESC(disable_brightness_adjust, "Disable brightness adjustment .");
 
1069 #ifdef CONFIG_FUJITSU_LAPTOP_DEBUG
 
1070 module_param_named(debug, dbg_level, uint, 0644);
 
1071 MODULE_PARM_DESC(debug, "Sets debug level bit-mask");
 
1074 MODULE_AUTHOR("Jonathan Woithe, Peter Gruber");
 
1075 MODULE_DESCRIPTION("Fujitsu laptop extras support");
 
1076 MODULE_VERSION(FUJITSU_DRIVER_VERSION);
 
1077 MODULE_LICENSE("GPL");
 
1080     ("dmi:*:svnFUJITSUSIEMENS:*:pvr:rvnFUJITSU:rnFJNB1D3:*:cvrS6410:*");
 
1082     ("dmi:*:svnFUJITSU:*:pvr:rvnFUJITSU:rnFJNB19C:*:cvrS7020:*");
 
1084 static struct pnp_device_id pnp_ids[] = {
 
1085         { .id = "FUJ02bf" },
 
1086         { .id = "FUJ02B1" },
 
1087         { .id = "FUJ02E3" },
 
1090 MODULE_DEVICE_TABLE(pnp, pnp_ids);