Greetings, Ive added polldev now and the driver is mostly finished. My questions : The old driver used HZ/20 as timing frequency should I simply translate this to ms instead?
It needs some minor cleanup, but feels good. Anything I've missed? Best wishes Kristoffer diff --git a/drivers/input/keyboard/jornada680_kbd.c b/drivers/input/keyboard/jornada680_kbd.c new file mode 100644 index 0000000..21bdc4b --- /dev/null +++ b/drivers/input/keyboard/jornada680_kbd.c @@ -0,0 +1,252 @@ +/* + * drivers/input/keyboard/jornada680_keyb.c + * + * HP Jornada 680/690 Scan keyboard + * + * Copyright (C) 2007 Kristoffer Ericson <[EMAIL PROTECTED]> + * + * Based on jornada680_keyb.c + * + * + * + * + * + */ + +#include <linux/input.h> +#include <linux/kernel.h> +#include <linux/module.h> +#include <linux/init.h> +#include <linux/jiffies.h> +#include <linux/platform_device.h> +#include <linux/interrupt.h> + +#include <asm/delay.h> +#include <asm/io.h> + +#define SCANHZ (HZ/20) + +#define PCCR 0xa4000104 +#define PDCR 0xa4000106 +#define PECR 0xa4000108 +#define PFCR 0xa400010a +#define PCDR 0xa4000124 +#define PDDR 0xa4000126 +#define PEDR 0xa4000128 +#define PFDR 0xa400012a +#define PGDR 0xa400012c +#define PHDR 0xa400012e +#define PJDR 0xa4000130 +#define PKDR 0xa4000132 +#define PLDR 0xa4000134 + +static unsigned char jornada_scancodes[] = { +/* PTD1 */ KEY_CAPSLOCK, KEY_MACRO, KEY_LEFTCTRL, 0, KEY_ESC, 0, 0, 0, KEY_F1, KEY_F2, + KEY_F3, KEY_F8, KEY_F7, KEY_F2, KEY_F4, KEY_F5, +/* PTD5 */ KEY_SLASH, KEY_APOSTROPHE, KEY_ENTER, 0, KEY_Z, 0, 0, 0, KEY_X, KEY_C, KEY_V, + KEY_DOT, KEY_COMMA, KEY_M, KEY_B, KEY_N, +/* PTD7 */ KEY_KP2, KEY_KP6, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, KEY_KP4, 0, 0, KEY_LEFTALT, + KEY_HANJA, +/* PTE0 */ 0, 0, 0, 0, KEY_FINANCE, 0, 0, 0, KEY_LEFTCTRL, 0, KEY_SPACE, KEY_KPDOT, + KEY_VOLUMEUP, 249, 0, 0, +/* PTE1 */ KEY_SEMICOLON, KEY_RIGHTBRACE, KEY_BACKSLASH, 0, KEY_A, 0, 0, 0, KEY_S, KEY_D, + KEY_F, KEY_L, KEY_K, KEY_J, KEY_G, KEY_H, +/* PTE3 */ KEY_KP8, KEY_LEFTMETA, KEY_RIGHTSHIFT, 0, KEY_TAB, 0, 0, + 0, 0, KEY_LEFTSHIFT, 0, 0, 0, 0, 0, 0, +/* PTE6 */ KEY_P, KEY_LEFTBRACE, KEY_BACKSPACE, 0, KEY_Q, 0, 0, + 0, KEY_W, KEY_E, KEY_R, KEY_O, KEY_I, KEY_U, KEY_T, KEY_R, +/* PTE7 */ KEY_0, KEY_MINUS, KEY_EQUAL, 0, KEY_1, 0, 0, + 0, KEY_2, KEY_3, KEY_4, KEY_9, KEY_8, KEY_7, KEY_5, KEY_6, +/* **** */ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0 +}; + +struct jornadakbd { + unsigned char jornada_keymap[ARRAY_SIZE(jornada_scancodes)]; + struct input_polled_dev *poll_dev; + unsigned char length; + unsigned char s0[16], s1[16]; +}; + +unsigned int scan_jiffies; + +void jornada_check_kbd(struct jornadakbd *jornadakbd, unsigned char *new, unsigned char *old) +{ + const unsigned char *table = jornadakbd->jornada_keymap; + unsigned char length = jornadakbd->length; + unsigned int bit, xor; + + while (length-- > 0) { + if ((xor = *new ^ *old) == 0) + table += 8; + else { + for(bit = 0x01; bit < 0x100;bit <<= 1) { + if (xor & bit) { + input_report_key(jornadakbd->poll_dev, *table, !(*new & bit)); + input_sync(jornadakbd->poll_dev); + } + } + } + new++; + old++; + } +} + +int jornada_scan_keyb(unsigned char *s) +{ + int i; + unsigned short ec_static, dc_static; /* = UINT16_t */ + unsigned char matrix_switch[] = { + 0xfd, 0xff, /* PTD1 PD(1) */ + 0xdf, 0xff, /* PTD5 PD(5) */ + 0x7f, 0xff, /* PTD7 PD(7) */ + 0xff, 0xfe, /* PTE0 PE(0) */ + 0xff, 0xfd, /* PTE1 PE(1) */ + 0xff, 0xf7, /* PTE3 PE(3) */ + 0xff, 0xbf, /* PTE6 PE(6) */ + 0xff, 0x7f, /* PTE7 PE(7) */ + }, *t = matrix_switch; + /* PD(x) : + 1. 0xcc0c & (1~(1 << (2*(x)+1))))) + 2. (0xf0cf & 0xfffff) */ + /* PE(x) : + 1. 0xcc0c & 0xffff + 2. 0xf0cf & (1~(1 << (2*(x)+1))))) */ + unsigned short matrix_PDE[] = { + 0xcc04, 0xf0cf, /* PD(1) */ + 0xc40c, 0xf0cf, /* PD(5) */ + 0x4c0c, 0xf0cf, /* PD(7) */ + 0xcc0c, 0xf0cd, /* PE(0) */ + 0xcc0c, 0xf0c7, /* PE(1) */ + 0xcc0c, 0xf04f, /* PE(3) */ + 0xcc0c, 0xd0cf, /* PE(6) */ + 0xcc0c, 0x70cf, /* PE(7) */ + }, *y = matrix_PDE; + /* Save these control reg bits */ + dc_static = (ctrl_inw(PDCR) & (~0xcc0c)); + ec_static = (ctrl_inw(PECR) & (~0xf0cf)); + + for (i = 0; i < 8; i++) { + /* disable output for all but the one we want to scan */ + ctrl_outw((dc_static | *y++), PDCR); + ctrl_outw((ec_static | *y++), PECR); + udelay(5); + + /* Get scanline row */ + ctrl_outb(*t++, PDDR); + ctrl_outb(*t++, PEDR); + udelay(50); + + /* Read data */ + *s++ = ctrl_inb(PCDR); + *s++ = ctrl_inb(PFDR); + } + /* Scan no lines */ + ctrl_outb(0xff, PDDR); + ctrl_outb(0xff, PEDR); + + /* Enable all scanlines */ + ctrl_outw((dc_static | (0x5555 & 0xcc0c)),PDCR); + ctrl_outw((ec_static | (0x5555 & 0xf0cf)),PECR); + /* Ignore extra keys and events */ + *s++ = ctrl_inb(PGDR); + *s++ = ctrl_inb(PHDR); + + return 0; +} + +static void jornadakbd680_poll(struct input_polled_dev *dev) +{ + struct jornadakbd *jornadakbd = dev->private; + scan_jiffies++; + + if (scan_jiffies & 1) { + if (!jornada_scan_keyb(jornadakbd->s0)) + jornada_check_kbd(jornadakbd, jornadakbd->s0, jornadakbd->s1); + else + memcpy(jornadakbd->s0, jornadakbd->s1, jornadakbd->length); + } else { + if (!jornada_scan_keyb(jornadakbd->s1)) + jornada_check_kbd(jornadkbd, jornadakbd->s1, jornadakbd->s0); + else + memcpy(jornadakbd->s1, jornadakbd->s0, jornadakbd->length); + } +} + +static int __devinit jornada680kbd_probe(struct platform_device *pdev) +{ + struct jornadakbd *jornadakbd; + struct input_polled_dev *poll_dev; + struct input_dev *input_dev; + int i, error; + + jornadakbd = kzalloc(sizeof(struct jornadakbd), GFP_KERNEL); + if (!jornadakbd) + return -ENOMEM; + + poll_dev = input_allocate_polled_device(); + if (!poll_dev) + goto failed; + + platform_set_drvdata(pdev, jornadakbd); + + jornadakbd->poll_dev = poll_dev; + + memcpy(jornadakbd->jornada_keymap, jornada_scancodes, sizeof(jornadakbd->jornada_keymap)); + + poll_dev->private = jornadakbd; + poll_dev->poll = jornadakbd680_poll; + poll_dev->poll_interval = SCAN_INTERVALL; + + input_dev = poll_dev->input; + input_dev->evbit[0] = BIT(EV_KEY) | BIT(EV_REP); + input_dev->name = "HP Jornada 680 keyboard"; + input_dev->phys = "jornadakbd/input0"; + input_dev->keycode = jornadakbd->jornada_keymap; + input_dev->keycodesize = sizeof(unsigned char); + input_dev->keycodemax = ARRAY_SIZE(jornada_scancodes); + input_dev->dev.parent = &pdev->dev; + + for (i = 0; i < ARRAY_SIZE(jornada_scancodes); i++) + set_bit(jornadakbd->jornada_keymap[i], input_dev->keybit); + clear_bit(0, input_dev->keybit); + + error = input_register_polled_device(jornadakbd->poll_dev); + if (error) + goto failed; + + return 0; + +failed: + kfree(jornadakbd); + return -ENOMEM; + +} + +static int jornada680_kbd_remove(struct platform_device *pdev) +{ + struct jornadakbd *jornadakbd = platform_get_drvdata(pdev); + + input_unregister_polled_device(jornadakbd->poll_dev); + input_free_polled_device(jornadakbd->poll_dev); + kfree(jornadakbd); + return 0; +} + +static struct platform_driver jornada680_kbd_driver = { + .driver = { + .name = "jornada680_kbd", + }, + .probe = jornada680_kbd_probe, + .remove = jornada680_kbd_remove, +}; + +static int __init jornada680kbd_init(void) +{ + return platform_driver_register(&jornada680kbd_driver); +} + +static void __exit jornada680kbd_exit(void) +{ + platform_driver_unregister(&jornada680kbd_driver); +} On Mon, 30 Jul 2007 08:39:20 -0400 "Dmitry Torokhov" <[EMAIL PROTECTED]> wrote: > Hi Kristoffer, > > On 7/27/07, Kristoffer Ericson <[EMAIL PROTECTED]> wrote: > > Dmitry, > > > > Ive merged scan_keyb and started to minimize it for just one keyboard, just > > want to hear from you if this is an acceptable approach. > > Its a work in progress, just want to know if im on the right path. > > > > Like I said, please look at input-polldev > (drivers/input/input-polldev.c). It takes care of setting up workqueue > used for polling and will only poll if there are users so the driver > only needs to provide miplementation of scan function. Take a look at > drivers/input/keyboard/aaed2000_kbd.c to see how it is used. > > -- > Dmitry -- Kristoffer Ericson <[EMAIL PROTECTED]>