#include <linux/kdb.h>
#include <linux/keyboard.h>
#include <linux/ctype.h>
#include <linux/module.h>
#include <linux/io.h>
#define KBD_STATUS_REG 0x64
#define KBD_DATA_REG 0x60
#define KBD_STAT_OBF 0x01
#define KBD_STAT_MOUSE_OBF 0x20
static int kbd_exists;
int kdb_get_kbd_char(void)
{
int scancode, scanstatus;
static int shift_lock;
static int shift_key;
static int ctrl_key;
u_short keychar;
if (KDB_FLAG(NO_I8042) || KDB_FLAG(NO_VT_CONSOLE) ||
(inb(KBD_STATUS_REG) == 0xff && inb(KBD_DATA_REG) == 0xff)) {
kbd_exists = 0;
return -1;
}
kbd_exists = 1;
if ((inb(KBD_STATUS_REG) & KBD_STAT_OBF) == 0)
return -1;
scancode = inb(KBD_DATA_REG);
scanstatus = inb(KBD_STATUS_REG);
if (scanstatus & KBD_STAT_MOUSE_OBF)
return -1;
if (((scancode&0x7f) == 0x2a) || ((scancode&0x7f) == 0x36)) {
if ((scancode & 0x80) == 0)
shift_key = 1;
else
shift_key = 0;
return -1;
}
if ((scancode&0x7f) == 0x1d) {
if ((scancode & 0x80) == 0)
ctrl_key = 1;
else
ctrl_key = 0;
return -1;
}
if ((scancode & 0x80) != 0)
return -1;
scancode &= 0x7f;
if (scancode == 0x3a) {
shift_lock ^= 1;
#ifdef KDB_BLINK_LED
kdb_toggleled(0x4);
#endif
return -1;
}
if (scancode == 0x0e) {
return 8;
}
switch (scancode) {
case 0xF:
return 9;
case 0x53:
return 4;
case 0x47:
return 1;
case 0x4F:
return 5;
case 0x4B:
return 2;
case 0x48:
return 16;
case 0x50:
return 14;
case 0x4D:
return 6;
}
if (scancode == 0xe0)
return -1;
if (scancode == 0x73)
scancode = 0x59;
else if (scancode == 0x7d)
scancode = 0x7c;
if (!shift_lock && !shift_key && !ctrl_key) {
keychar = plain_map[scancode];
} else if ((shift_lock || shift_key) && key_maps[1]) {
keychar = key_maps[1][scancode];
} else if (ctrl_key && key_maps[4]) {
keychar = key_maps[4][scancode];
} else {
keychar = 0x0020;
kdb_printf("Unknown state/scancode (%d)\n", scancode);
}
keychar &= 0x0fff;
if (keychar == '\t')
keychar = ' ';
switch (KTYP(keychar)) {
case KT_LETTER:
case KT_LATIN:
if (isprint(keychar))
break;
case KT_SPEC:
if (keychar == K_ENTER)
break;
default:
return -1;
}
if ((scancode & 0x7f) == 0x1c) {
while ((inb(KBD_STATUS_REG) & KBD_STAT_OBF) == 0)
;
scancode = inb(KBD_DATA_REG);
scanstatus = inb(KBD_STATUS_REG);
while (scanstatus & KBD_STAT_MOUSE_OBF) {
scancode = inb(KBD_DATA_REG);
scanstatus = inb(KBD_STATUS_REG);
}
if (scancode != 0x9c) {
kdb_printf("kdb: expected enter got 0x%x status 0x%x\n",
scancode, scanstatus);
}
return 13;
}
return keychar & 0xff;
}
EXPORT_SYMBOL_GPL(kdb_get_kbd_char);