linux/drivers/staging/nvec/nvec_kbd.c
<<
>>
Prefs
   1/*
   2 * nvec_kbd: keyboard driver for a NVIDIA compliant embedded controller
   3 *
   4 * Copyright (C) 2011 The AC100 Kernel Team <ac100@lists.launchpad.net>
   5 *
   6 * Authors:  Pierre-Hugues Husson <phhusson@free.fr>
   7 *           Marc Dietrich <marvin24@gmx.de>
   8 *
   9 * This file is subject to the terms and conditions of the GNU General Public
  10 * License.  See the file "COPYING" in the main directory of this archive
  11 * for more details.
  12 *
  13 */
  14
  15#include <linux/module.h>
  16#include <linux/slab.h>
  17#include <linux/input.h>
  18#include <linux/delay.h>
  19#include <linux/platform_device.h>
  20
  21#include "nvec-keytable.h"
  22#include "nvec.h"
  23
  24enum kbd_subcmds {
  25        CNFG_WAKE = 3,
  26        CNFG_WAKE_KEY_REPORTING,
  27        SET_LEDS = 0xed,
  28        ENABLE_KBD = 0xf4,
  29        DISABLE_KBD,
  30};
  31
  32static unsigned char keycodes[ARRAY_SIZE(code_tab_102us)
  33                              + ARRAY_SIZE(extcode_tab_us102)];
  34
  35struct nvec_keys {
  36        struct input_dev *input;
  37        struct notifier_block notifier;
  38        struct nvec_chip *nvec;
  39        bool caps_lock;
  40};
  41
  42static struct nvec_keys keys_dev;
  43
  44static void nvec_kbd_toggle_led(void)
  45{
  46        char buf[] = { NVEC_KBD, SET_LEDS, 0 };
  47
  48        keys_dev.caps_lock = !keys_dev.caps_lock;
  49
  50        if (keys_dev.caps_lock)
  51                /* should be BIT(0) only, firmware bug? */
  52                buf[2] = BIT(0) | BIT(1) | BIT(2);
  53
  54        nvec_write_async(keys_dev.nvec, buf, sizeof(buf));
  55}
  56
  57static int nvec_keys_notifier(struct notifier_block *nb,
  58                              unsigned long event_type, void *data)
  59{
  60        int code, state;
  61        unsigned char *msg = (unsigned char *)data;
  62
  63        if (event_type == NVEC_KB_EVT) {
  64                int _size = (msg[0] & (3 << 5)) >> 5;
  65
  66/* power on/off button */
  67                if (_size == NVEC_VAR_SIZE)
  68                        return NOTIFY_STOP;
  69
  70                if (_size == NVEC_3BYTES)
  71                        msg++;
  72
  73                code = msg[1] & 0x7f;
  74                state = msg[1] & 0x80;
  75
  76                if (code_tabs[_size][code] == KEY_CAPSLOCK && state)
  77                        nvec_kbd_toggle_led();
  78
  79                input_report_key(keys_dev.input, code_tabs[_size][code],
  80                                 !state);
  81                input_sync(keys_dev.input);
  82
  83                return NOTIFY_STOP;
  84        }
  85
  86        return NOTIFY_DONE;
  87}
  88
  89static int nvec_kbd_event(struct input_dev *dev, unsigned int type,
  90                          unsigned int code, int value)
  91{
  92        struct nvec_chip *nvec = keys_dev.nvec;
  93        char buf[] = { NVEC_KBD, SET_LEDS, 0 };
  94
  95        if (type == EV_REP)
  96                return 0;
  97
  98        if (type != EV_LED)
  99                return -1;
 100
 101        if (code != LED_CAPSL)
 102                return -1;
 103
 104        buf[2] = !!value;
 105        nvec_write_async(nvec, buf, sizeof(buf));
 106
 107        return 0;
 108}
 109
 110static int nvec_kbd_probe(struct platform_device *pdev)
 111{
 112        struct nvec_chip *nvec = dev_get_drvdata(pdev->dev.parent);
 113        int i, j, err;
 114        struct input_dev *idev;
 115        char    clear_leds[] = { NVEC_KBD, SET_LEDS, 0 },
 116                enable_kbd[] = { NVEC_KBD, ENABLE_KBD },
 117                cnfg_wake[] = { NVEC_KBD, CNFG_WAKE, true, true },
 118                cnfg_wake_key_reporting[] = { NVEC_KBD, CNFG_WAKE_KEY_REPORTING,
 119                                                true };
 120
 121        j = 0;
 122
 123        for (i = 0; i < ARRAY_SIZE(code_tab_102us); ++i)
 124                keycodes[j++] = code_tab_102us[i];
 125
 126        for (i = 0; i < ARRAY_SIZE(extcode_tab_us102); ++i)
 127                keycodes[j++] = extcode_tab_us102[i];
 128
 129        idev = devm_input_allocate_device(&pdev->dev);
 130        idev->name = "nvec keyboard";
 131        idev->phys = "nvec";
 132        idev->evbit[0] = BIT_MASK(EV_KEY) | BIT_MASK(EV_REP) | BIT_MASK(EV_LED);
 133        idev->ledbit[0] = BIT_MASK(LED_CAPSL);
 134        idev->event = nvec_kbd_event;
 135        idev->keycode = keycodes;
 136        idev->keycodesize = sizeof(unsigned char);
 137        idev->keycodemax = ARRAY_SIZE(keycodes);
 138
 139        for (i = 0; i < ARRAY_SIZE(keycodes); ++i)
 140                set_bit(keycodes[i], idev->keybit);
 141
 142        clear_bit(0, idev->keybit);
 143        err = input_register_device(idev);
 144        if (err)
 145                return err;
 146
 147        keys_dev.input = idev;
 148        keys_dev.notifier.notifier_call = nvec_keys_notifier;
 149        keys_dev.nvec = nvec;
 150        nvec_register_notifier(nvec, &keys_dev.notifier, 0);
 151
 152        /* Enable keyboard */
 153        nvec_write_async(nvec, enable_kbd, 2);
 154
 155        /* configures wake on special keys */
 156        nvec_write_async(nvec, cnfg_wake, 4);
 157        /* enable wake key reporting */
 158        nvec_write_async(nvec, cnfg_wake_key_reporting, 3);
 159
 160        /* Disable caps lock LED */
 161        nvec_write_async(nvec, clear_leds, sizeof(clear_leds));
 162
 163        return 0;
 164}
 165
 166static int nvec_kbd_remove(struct platform_device *pdev)
 167{
 168        struct nvec_chip *nvec = dev_get_drvdata(pdev->dev.parent);
 169        char disable_kbd[] = { NVEC_KBD, DISABLE_KBD },
 170             uncnfg_wake_key_reporting[] = { NVEC_KBD, CNFG_WAKE_KEY_REPORTING,
 171                                                false };
 172        nvec_write_async(nvec, uncnfg_wake_key_reporting, 3);
 173        nvec_write_async(nvec, disable_kbd, 2);
 174        nvec_unregister_notifier(nvec, &keys_dev.notifier);
 175
 176        return 0;
 177}
 178
 179static struct platform_driver nvec_kbd_driver = {
 180        .probe  = nvec_kbd_probe,
 181        .remove = nvec_kbd_remove,
 182        .driver = {
 183                .name = "nvec-kbd",
 184                .owner = THIS_MODULE,
 185        },
 186};
 187
 188module_platform_driver(nvec_kbd_driver);
 189
 190MODULE_AUTHOR("Marc Dietrich <marvin24@gmx.de>");
 191MODULE_DESCRIPTION("NVEC keyboard driver");
 192MODULE_ALIAS("platform:nvec-kbd");
 193MODULE_LICENSE("GPL");
 194