/* * linux/kernel/chr_drv/keyboard.c * * Keyboard driver for Linux v0.99 using Latin-1. * * Written for linux by Johan Myreen as a translation from * the assembly version by Linus (with diacriticals added) * * Some additional features added by Christoph Niemann (ChN), March 1993 * Loadable keymaps by Risto Kankkunen, May 1993 * Diacriticals redone & other small changes, aeb@cwi.nl, June 1993 */ #define KEYBOARD_IRQ 1 #include #include #include #include #include #include #include #include #include #include "kbd_kern.h" #include "diacr.h" #define SIZE(x) (sizeof(x)/sizeof((x)[0])) #ifndef KBD_DEFMODE #define KBD_DEFMODE ((1 << VC_REPEAT) | (1 << VC_META)) #endif #ifndef KBD_DEFLEDS #define KBD_DEFLEDS (1 << VC_NUMLOCK) #endif #ifndef KBD_DEFLOCK #define KBD_DEFLOCK 0 #endif /* * The default IO slowdown is doing 'inb()'s from 0x61, which should be * safe. But as that is the keyboard controller chip address, we do our * slowdowns here by doing short jumps: the keyboard controller should * be able to keep up */ #define REALLY_SLOW_IO #define SLOW_IO_BY_JUMPING #include #include extern void do_keyboard_interrupt(void); extern void ctrl_alt_del(void); extern void change_console(unsigned int new_console); extern void scrollback(int); extern void scrollfront(int); #define fake_keyboard_interrupt() \ __asm__ __volatile__("int $0x21") unsigned char kbd_read_mask = 0x01; /* modified by psaux.c */ /* * global state includes the following, and various static variables * in this module: prev_scancode, shift_state, diacr, npadch, * dead_key_next, last_console */ /* shift state counters.. */ static unsigned char k_down[NR_SHIFT] = {0, }; /* keyboard key bitmap */ static unsigned long key_down[8] = { 0, }; static int want_console = -1; static int last_console = 0; /* last used VC */ static int dead_key_next = 0; static int shift_state = 0; static int npadch = -1; /* -1 or number assembled on pad */ static unsigned char diacr = 0; static char rep = 0; /* flag telling character repeat */ struct kbd_struct kbd_table[NR_CONSOLES]; static struct kbd_struct * kbd = kbd_table; static struct tty_struct * tty = NULL; /* used only by send_data - set by keyboard_interrupt */ static volatile unsigned char acknowledge = 0; static volatile unsigned char resend = 0; typedef void (*k_hand)(unsigned char value, char up_flag); typedef void (k_handfn)(unsigned char value, char up_flag); static k_handfn do_self, do_fn, do_spec, do_pad, do_dead, do_cons, do_cur, do_shift, do_meta, do_ascii, do_lock, do_lowercase; static k_hand key_handler[] = { do_self, do_fn, do_spec, do_pad, do_dead, do_cons, do_cur, do_shift, do_meta, do_ascii, do_lock, do_lowercase }; /* maximum values each key_handler can handle */ const int max_vals[] = { 255, NR_FUNC - 1, 14, 17, 4, 255, 3, NR_SHIFT, 255, 9, 3, 255 }; const int NR_TYPES = SIZE(max_vals); static void put_queue(int); static unsigned char handle_diacr(unsigned char); /* pt_regs - set by keyboard_interrupt(), used by show_ptregs() */ static struct pt_regs * pt_regs; static int got_break = 0; static inline void kb_wait(void) { int i; for (i=0; i<0x10000; i++) if ((inb_p(0x64) & 0x02) == 0) break; } static inline void send_cmd(unsigned char c) { kb_wait(); outb(c,0x64); } /* * Translation of escaped scancodes to keysyms. * This should be user-settable. */ #define E0_BASE 96 #define E0_KPENTER (E0_BASE+0) #define E0_RCTRL (E0_BASE+1) #define E0_KPSLASH (E0_BASE+2) #define E0_PRSCR (E0_BASE+3) #define E0_RALT (E0_BASE+4) #define E0_BREAK (E0_BASE+5) /* (control-pause) */ #define E0_HOME (E0_BASE+6) #define E0_UP (E0_BASE+7) #define E0_PGUP (E0_BASE+8) #define E0_LEFT (E0_BASE+9) #define E0_RIGHT (E0_BASE+10) #define E0_END (E0_BASE+11) #define E0_DOWN (E0_BASE+12) #define E0_PGDN (E0_BASE+13) #define E0_INS (E0_BASE+14) #define E0_DEL (E0_BASE+15) /* BTC */ #define E0_MACRO (E0_BASE+16) /* LK450 */ #define E0_F13 (E0_BASE+17) #define E0_F14 (E0_BASE+18) #define E0_HELP (E0_BASE+19) #define E0_DO (E0_BASE+20) #define E0_F17 (E0_BASE+21) #define E0_KPMINPLUS (E0_BASE+22) #define E1_PAUSE (E0_BASE+23) static unsigned char e0_keys[128] = { 0, 0, 0, 0, 0, 0, 0, 0, /* 0x00-0x07 */ 0, 0, 0, 0, 0, 0, 0, 0, /* 0x08-0x0f */ 0, 0, 0, 0, 0, 0, 0, 0, /* 0x10-0x17 */ 0, 0, 0, 0, E0_KPENTER, E0_RCTRL, 0, 0, /* 0x18-0x1f */ 0, 0, 0, 0, 0, 0, 0, 0, /* 0x20-0x27 */ 0, 0, 0, 0, 0, 0, 0, 0, /* 0x28-0x2f */ 0, 0, 0, 0, 0, E0_KPSLASH, 0, E0_PRSCR, /* 0x30-0x37 */ E0_RALT, 0, 0, 0, 0, E0_F13, E0_F14, E0_HELP, /* 0x38-0x3f */ E0_DO, E0_F17, 0, 0, 0, 0, E0_BREAK, E0_HOME, /* 0x40-0x47 */ E0_UP, E0_PGUP, 0, E0_LEFT, 0, E0_RIGHT, E0_KPMINPLUS, E0_END,/* 0x48-0x4f */ E0_DOWN, E0_PGDN, E0_INS, E0_DEL, 0, 0, 0, 0, /* 0x50-0x57 */ 0, 0, 0, 0, 0, 0, 0, 0, /* 0x58-0x5f */ 0, 0, 0, 0, 0, 0, 0, 0, /* 0x60-0x67 */ 0, 0, 0, 0, 0, 0, 0, E0_MACRO, /* 0x68-0x6f */ 0, 0, 0, 0, 0, 0, 0, 0, /* 0x70-0x77 */ 0, 0, 0, 0, 0, 0, 0, 0 /* 0x78-0x7f */ }; static void keyboard_interrupt(int int_pt_regs) { unsigned char scancode; static unsigned int prev_scancode = 0; /* remember E0, E1 */ char up_flag; /* 0 or 0200 */ char raw_mode; pt_regs = (struct pt_regs *) int_pt_regs; send_cmd(0xAD); /* disable keyboard */ kb_wait(); if ((inb_p(0x64) & kbd_read_mask) != 0x01) goto end_kbd_intr; scancode = inb(0x60); mark_bh(KEYBOARD_BH); if (scancode == 0xfa) { acknowledge = 1; goto end_kbd_intr; } else if (scancode == 0xfe) { resend = 1; goto end_kbd_intr; } tty = TTY_TABLE(0); kbd = kbd_table + fg_console; if ((raw_mode = vc_kbd_mode(kbd,VC_RAW))) { put_queue(scancode); /* we do not return yet, because we want to maintain the key_down array, so that we have the correct values when finishing RAW mode or when changing VT's */ } if (scancode == 0xe0 || scancode == 0xe1) { prev_scancode = scancode; goto end_kbd_intr; } /* * Convert scancode to keysym, using prev_scancode. */ up_flag = (scancode & 0200); scancode &= 0x7f; if (prev_scancode) { /* * usually it will be 0xe0, but a Pause key generates * e1 1d 45 e1 9d c5 when pressed, and nothing when released */ if (prev_scancode != 0xe0) { if (prev_scancode == 0xe1 && scancode == 0x1d) { prev_scancode = 0x100; goto end_kbd_intr; } else if (prev_scancode == 0x100 && scancode == 0x45) { scancode = E1_PAUSE; prev_scancode = 0; } else { printk("keyboard: unknown e1 escape sequence\n"); prev_scancode = 0; goto end_kbd_intr; } } else { prev_scancode = 0; /* * The keyboard maintains its own internal caps lock and * num lock statuses. In caps lock mode E0 AA precedes make * code and E0 2A follows break code. In num lock mode, * E0 2A precedes make code and E0 AA follows break code. * We do our own book-keeping, so we will just ignore these. */ /* * For my keyboard there is no caps lock mode, but there are * both Shift-L and Shift-R modes. The former mode generates * E0 2A / E0 AA pairs, the latter E0 B6 / E0 36 pairs. * So, we should also ignore the latter. - aeb@cwi.nl */ if (scancode == 0x2a || scancode == 0x36) goto end_kbd_intr; if (e0_keys[scancode]) scancode = e0_keys[scancode]; else if (!raw_mode) { printk("keyboard: unknown scancode e0 %02x\n", scancode); goto end_kbd_intr; } } } else if (scancode >= E0_BASE && !raw_mode) { printk("keyboard: scancode (%02x) not in range 00 - %2x\n", scancode, E0_BASE - 1); goto end_kbd_intr; } /* * At this point the variable `scancode' contains the keysym. * We keep track of the up/down status of the key, and * return the keysym if in MEDIUMRAW mode. * (Note: earlier kernels had a bug and did not pass the up/down * bit to applications.) */ if (up_flag) { clear_bit(scancode, key_down); rep = 0; } else rep = set_bit(scancode, key_down); if (raw_mode) goto end_kbd_intr; if (vc_kbd_mode(kbd, VC_MEDIUMRAW)) { put_queue(scancode + up_flag); goto end_kbd_intr; } /* * Small change in philosophy: earlier we defined repetition by * rep = scancode == prev_keysym; * prev_keysym = scancode; * but now by the fact that the depressed key was down already. * Does this ever make a difference? */ /* * Repeat a key only if the input buffers are empty or the * characters get echoed locally. This makes key repeat usable * with slow applications and under heavy loads. */ if (!rep || (vc_kbd_mode(kbd,VC_REPEAT) && tty && (L_ECHO(tty) || (EMPTY(&tty->secondary) && EMPTY(&tty->read_q))))) { u_short key_code; u_char type; /* the XOR below used to be an OR */ int shift_final = shift_state ^ kbd->lockstate; key_code = key_map[shift_final][scancode]; type = KTYP(key_code); if (type == KT_LETTER) { type = KT_LATIN; if (vc_kbd_led(kbd,VC_CAPSLOCK)) key_code = key_map[shift_final ^ (1<read_q; if (LEFT(qp)) { qp->buf[qp->head] = ch; INC(qp->head); } } static void puts_queue(char *cp) { struct tty_queue *qp; char ch; /* why interruptible here, plain wake_up above? */ wake_up_interruptible(&keypress_wait); if (!tty) return; qp = &tty->read_q; while ((ch = *(cp++)) != 0) { if (LEFT(qp)) { qp->buf[qp->head] = ch; INC(qp->head); } } } static void applkey(int key, char mode) { static char buf[] = { 0x1b, 'O', 0x00, 0x00 }; buf[1] = (mode ? 'O' : '['); buf[2] = key; puts_queue(buf); } static void enter(void) { put_queue(13); if (vc_kbd_mode(kbd,VC_CRLF)) put_queue(10); } static void caps_toggle(void) { if (rep) return; chg_vc_kbd_led(kbd,VC_CAPSLOCK); } static void caps_on(void) { if (rep) return; set_vc_kbd_led(kbd,VC_CAPSLOCK); } static void show_ptregs(void) { if (!pt_regs) return; printk("\n"); printk("EIP: %04x:%08lx",0xffff & pt_regs->cs,pt_regs->eip); if (pt_regs->cs & 3) printk(" ESP: %04x:%08lx",0xffff & pt_regs->ss,pt_regs->esp); printk(" EFLAGS: %08lx\n",pt_regs->eflags); printk("EAX: %08lx EBX: %08lx ECX: %08lx EDX: %08lx\n", pt_regs->orig_eax,pt_regs->ebx,pt_regs->ecx,pt_regs->edx); printk("ESI: %08lx EDI: %08lx EBP: %08lx", pt_regs->esi, pt_regs->edi, pt_regs->ebp); printk(" DS: %04x ES: %04x FS: %04x GS: %04x\n", 0xffff & pt_regs->ds,0xffff & pt_regs->es, 0xffff & pt_regs->fs,0xffff & pt_regs->gs); } static void hold(void) { if (rep || !tty) return; /* * Note: SCROLLOCK wil be set (cleared) by stop_tty (start_tty); * these routines are also activated by ^S/^Q. * (And SCROLLOCK can also be set by the ioctl KDSETLED.) */ if (tty->stopped) start_tty(tty); else stop_tty(tty); } #if 0 /* unused at present - and the VC_PAUSE bit is not used anywhere either */ static void pause(void) { chg_vc_kbd_mode(kbd,VC_PAUSE); } #endif static void num(void) { if (vc_kbd_mode(kbd,VC_APPLIC)) { applkey('P', 1); return; } if (!rep) /* no autorepeat for numlock, ChN */ chg_vc_kbd_led(kbd,VC_NUMLOCK); } static void lastcons(void) { /* pressing alt-printscreen switches to the last used console, ChN */ want_console = last_console; } static void send_intr(void) { got_break = 1; } static void scrll_forw(void) { scrollfront(0); } static void scrll_back(void) { scrollback(0); } static void boot_it(void) { ctrl_alt_del(); } static void compose(void) { dead_key_next = 1; } static void do_spec(unsigned char value, char up_flag) { typedef void (*fnp)(void); fnp fn_table[] = { NULL, enter, show_ptregs, show_mem, show_state, send_intr, lastcons, caps_toggle, num, hold, scrll_forw, scrll_back, boot_it, caps_on, compose }; if (up_flag) return; if (value >= SIZE(fn_table)) return; if (!fn_table[value]) return; fn_table[value](); } static void do_lowercase(unsigned char value, char up_flag) { printk("keyboard.c: do_lowercase was called - impossible\n"); } static void do_self(unsigned char value, char up_flag) { if (up_flag) return; /* no action, if this is a key release */ if (diacr) value = handle_diacr(value); if (dead_key_next) { dead_key_next = 0; diacr = value; return; } put_queue(value); } #define A_GRAVE '`' #define A_ACUTE '\'' #define A_CFLEX '^' #define A_TILDE '~' #define A_DIAER '"' static unsigned char ret_diacr[] = {A_GRAVE, A_ACUTE, A_CFLEX, A_TILDE, A_DIAER }; /* If a dead key pressed twice, output a character corresponding to it, */ /* otherwise just remember the dead key. */ static void do_dead(unsigned char value, char up_flag) { if (up_flag) return; value = ret_diacr[value]; if (diacr == value) { /* pressed twice */ diacr = 0; put_queue(value); return; } diacr = value; } /* If space is pressed, return the character corresponding the pending */ /* dead key, otherwise try to combine the two. */ unsigned char handle_diacr(unsigned char ch) { int d = diacr; int i; diacr = 0; if (ch == ' ') return d; for (i = 0; i < accent_table_size; i++) if(accent_table[i].diacr == d && accent_table[i].base == ch) return accent_table[i].result; put_queue(d); return ch; } static void do_cons(unsigned char value, char up_flag) { if (up_flag) return; want_console = value; } static void do_fn(unsigned char value, char up_flag) { if (up_flag) return; if (value < SIZE(func_table)) puts_queue(func_table[value]); else printk("do_fn called with value=%d\n", value); } static void do_pad(unsigned char value, char up_flag) { static char *pad_chars = "0123456789+-*/\015,.?"; static char *app_map = "pqrstuvwxylSRQMnn?"; if (up_flag) return; /* no action, if this is a key release */ /* kludge... shift forces cursor/number keys */ if (vc_kbd_mode(kbd,VC_APPLIC) && !k_down[KG_SHIFT]) { applkey(app_map[value], 1); return; } if (!vc_kbd_led(kbd,VC_NUMLOCK)) switch (value) { case KVAL(K_PCOMMA): case KVAL(K_PDOT): do_fn(KVAL(K_REMOVE), 0); return; case KVAL(K_P0): do_fn(KVAL(K_INSERT), 0); return; case KVAL(K_P1): do_fn(KVAL(K_SELECT), 0); return; case KVAL(K_P2): do_cur(KVAL(K_DOWN), 0); return; case KVAL(K_P3): do_fn(KVAL(K_PGDN), 0); return; case KVAL(K_P4): do_cur(KVAL(K_LEFT), 0); return; case KVAL(K_P6): do_cur(KVAL(K_RIGHT), 0); return; case KVAL(K_P7): do_fn(KVAL(K_FIND), 0); return; case KVAL(K_P8): do_cur(KVAL(K_UP), 0); return; case KVAL(K_P9): do_fn(KVAL(K_PGUP), 0); return; case KVAL(K_P5): applkey('G', vc_kbd_mode(kbd, VC_APPLIC)); return; } put_queue(pad_chars[value]); if (value == KVAL(K_PENTER) && vc_kbd_mode(kbd, VC_CRLF)) put_queue(10); } static void do_cur(unsigned char value, char up_flag) { static char *cur_chars = "BDCA"; if (up_flag) return; applkey(cur_chars[value], vc_kbd_mode(kbd,VC_CKMODE)); } static void do_shift(unsigned char value, char up_flag) { int old_state = shift_state; if (rep) return; /* kludge... */ if (value == KVAL(K_CAPSSHIFT)) { value = KVAL(K_SHIFT); clr_vc_kbd_led(kbd, VC_CAPSLOCK); } if (up_flag) { /* handle the case that two shift or control keys are depressed simultaneously */ if (k_down[value]) k_down[value]--; } else k_down[value]++; if (k_down[value]) shift_state |= (1 << value); else shift_state &= ~ (1 << value); /* kludge */ if (up_flag && shift_state != old_state && npadch != -1) { put_queue(npadch); npadch = -1; } } /* called after returning from RAW mode or when changing consoles - recompute k_down[] and shift_state from key_down[] */ void compute_shiftstate(void) { int i, j, k, sym, val; shift_state = 0; for(i=0; i < SIZE(k_down); i++) k_down[i] = 0; for(i=0; i < SIZE(key_down); i++) if(key_down[i]) { /* skip this word if not a single bit on */ k = (i<<5); for(j=0; j<32; j++,k++) if(test_bit(k, key_down)) { sym = key_map[0][k]; if(KTYP(sym) == KT_SHIFT) { val = KVAL(sym); k_down[val]++; shift_state |= (1< 0); return 0; } /* * This routine is the bottom half of the keyboard interrupt * routine, and runs with all interrupts enabled. It does * console changing, led setting and copy_to_cooked, which can * take a reasonably long time. * * Aside from timing (which isn't really that important for * keyboard interrupts as they happen often), using the software * interrupt routines for this thing allows us to easily mask * this when we don't want any of the above to happen. Not yet * used, but this allows for easy and efficient race-condition * prevention later on. */ static void kbd_bh(void * unused) { static unsigned char old_leds = 0xff; unsigned char leds = kbd_table[fg_console].ledstate; if (leds != old_leds) { old_leds = leds; if (!send_data(0xed) || !send_data(leds)) send_data(0xf4); /* re-enable kbd if any errors */ } if (want_console >= 0) { if (want_console != fg_console) { last_console = fg_console; change_console(want_console); } want_console = -1; } if (got_break) { if (tty && !I_IGNBRK(tty)) { if (I_BRKINT(tty)) { flush_input(tty); flush_output(tty); if (tty->pgrp > 0) kill_pg(tty->pgrp, SIGINT, 1); } else { cli(); if (LEFT(&tty->read_q) >= 2) { set_bit(tty->read_q.head, &tty->readq_flags); put_queue(TTY_BREAK); put_queue(0); } sti(); } } got_break = 0; } do_keyboard_interrupt(); cli(); if ((inb_p(0x64) & kbd_read_mask) == 0x01) fake_keyboard_interrupt(); sti(); } long no_idt[2] = {0, 0}; /* * This routine reboots the machine by asking the keyboard * controller to pulse the reset-line low. We try that for a while, * and if it doesn't work, we do some other stupid things. */ void hard_reset_now(void) { int i, j; extern unsigned long pg0[1024]; sti(); /* rebooting needs to touch the page at absolute addr 0 */ pg0[0] = 7; *((unsigned short *)0x472) = 0x1234; for (;;) { for (i=0; i<100; i++) { kb_wait(); for(j = 0; j < 100000 ; j++) /* nothing */; outb(0xfe,0x64); /* pulse reset low */ } __asm__("\tlidt _no_idt"); } } unsigned long kbd_init(unsigned long kmem_start) { int i; struct kbd_struct * kbd; kbd = kbd_table + 0; for (i = 0 ; i < NR_CONSOLES ; i++,kbd++) { kbd->ledstate = KBD_DEFLEDS; kbd->default_ledstate = KBD_DEFLEDS; kbd->lockstate = KBD_DEFLOCK; kbd->modeflags = KBD_DEFMODE; } bh_base[KEYBOARD_BH].routine = kbd_bh; request_irq(KEYBOARD_IRQ,keyboard_interrupt); mark_bh(KEYBOARD_BH); return kmem_start; }