| /*********************************************************************** |
| * |
| * (C) Copyright 2004 |
| * DENX Software Engineering |
| * Wolfgang Denk, wd@denx.de |
| * All rights reserved. |
| * |
| * PS/2 keyboard driver |
| * |
| * Originally from linux source (drivers/char/pc_keyb.c) |
| * |
| ***********************************************************************/ |
| |
| #include <common.h> |
| |
| #ifdef CONFIG_PS2KBD |
| |
| #include <keyboard.h> |
| #include <pc_keyb.h> |
| |
| #undef KBG_DEBUG |
| |
| #ifdef KBG_DEBUG |
| #define PRINTF(fmt,args...) printf (fmt ,##args) |
| #else |
| #define PRINTF(fmt,args...) |
| #endif |
| |
| |
| /* |
| * This reads the keyboard status port, and does the |
| * appropriate action. |
| * |
| */ |
| static unsigned char handle_kbd_event(void) |
| { |
| unsigned char status = kbd_read_status(); |
| unsigned int work = 10000; |
| |
| while ((--work > 0) && (status & KBD_STAT_OBF)) { |
| unsigned char scancode; |
| |
| scancode = kbd_read_input(); |
| |
| /* Error bytes must be ignored to make the |
| Synaptics touchpads compaq use work */ |
| /* Ignore error bytes */ |
| if (!(status & (KBD_STAT_GTO | KBD_STAT_PERR))) { |
| if (status & KBD_STAT_MOUSE_OBF) |
| ; /* not supported: handle_mouse_event(scancode); */ |
| else |
| handle_scancode(scancode); |
| } |
| status = kbd_read_status(); |
| } |
| if (!work) |
| PRINTF("pc_keyb: controller jammed (0x%02X).\n", status); |
| return status; |
| } |
| |
| |
| static int kbd_read_data(void) |
| { |
| int val; |
| unsigned char status; |
| |
| val=-1; |
| status = kbd_read_status(); |
| if (status & KBD_STAT_OBF) { |
| val = kbd_read_input(); |
| if (status & (KBD_STAT_GTO | KBD_STAT_PERR)) |
| val = -2; |
| } |
| return val; |
| } |
| |
| static int kbd_wait_for_input(void) |
| { |
| unsigned long timeout; |
| int val; |
| |
| timeout = KBD_TIMEOUT; |
| val=kbd_read_data(); |
| while(val < 0) { |
| if(timeout--==0) |
| return -1; |
| udelay(1000); |
| val=kbd_read_data(); |
| } |
| return val; |
| } |
| |
| |
| static int kb_wait(void) |
| { |
| unsigned long timeout = KBC_TIMEOUT * 10; |
| |
| do { |
| unsigned char status = handle_kbd_event(); |
| if (!(status & KBD_STAT_IBF)) |
| return 0; /* ok */ |
| udelay(1000); |
| timeout--; |
| } while (timeout); |
| return 1; |
| } |
| |
| static void kbd_write_command_w(int data) |
| { |
| if(kb_wait()) |
| PRINTF("timeout in kbd_write_command_w\n"); |
| kbd_write_command(data); |
| } |
| |
| static void kbd_write_output_w(int data) |
| { |
| if(kb_wait()) |
| PRINTF("timeout in kbd_write_output_w\n"); |
| kbd_write_output(data); |
| } |
| |
| static void kbd_send_data(unsigned char data) |
| { |
| kbd_write_output_w(data); |
| kbd_wait_for_input(); |
| } |
| |
| |
| static char * kbd_initialize(void) |
| { |
| int status; |
| |
| /* |
| * Test the keyboard interface. |
| * This seems to be the only way to get it going. |
| * If the test is successful a x55 is placed in the input buffer. |
| */ |
| kbd_write_command_w(KBD_CCMD_SELF_TEST); |
| if (kbd_wait_for_input() != 0x55) |
| return "Kbd: failed self test"; |
| /* |
| * Perform a keyboard interface test. This causes the controller |
| * to test the keyboard clock and data lines. The results of the |
| * test are placed in the input buffer. |
| */ |
| kbd_write_command_w(KBD_CCMD_KBD_TEST); |
| if (kbd_wait_for_input() != 0x00) |
| return "Kbd: interface failed self test"; |
| /* |
| * Enable the keyboard by allowing the keyboard clock to run. |
| */ |
| kbd_write_command_w(KBD_CCMD_KBD_ENABLE); |
| |
| /* |
| * Reset keyboard. If the read times out |
| * then the assumption is that no keyboard is |
| * plugged into the machine. |
| * This defaults the keyboard to scan-code set 2. |
| * |
| * Set up to try again if the keyboard asks for RESEND. |
| */ |
| do { |
| kbd_write_output_w(KBD_CMD_RESET); |
| status = kbd_wait_for_input(); |
| if (status == KBD_REPLY_ACK) |
| break; |
| if (status != KBD_REPLY_RESEND) { |
| PRINTF("status: %X\n",status); |
| return "Kbd: reset failed, no ACK"; |
| } |
| } while (1); |
| if (kbd_wait_for_input() != KBD_REPLY_POR) |
| return "Kbd: reset failed, no POR"; |
| |
| /* |
| * Set keyboard controller mode. During this, the keyboard should be |
| * in the disabled state. |
| * |
| * Set up to try again if the keyboard asks for RESEND. |
| */ |
| do { |
| kbd_write_output_w(KBD_CMD_DISABLE); |
| status = kbd_wait_for_input(); |
| if (status == KBD_REPLY_ACK) |
| break; |
| if (status != KBD_REPLY_RESEND) |
| return "Kbd: disable keyboard: no ACK"; |
| } while (1); |
| |
| kbd_write_command_w(KBD_CCMD_WRITE_MODE); |
| kbd_write_output_w(KBD_MODE_KBD_INT |
| | KBD_MODE_SYS |
| | KBD_MODE_DISABLE_MOUSE |
| | KBD_MODE_KCC); |
| |
| /* AMCC powerpc portables need this to use scan-code set 1 -- Cort */ |
| kbd_write_command_w(KBD_CCMD_READ_MODE); |
| if (!(kbd_wait_for_input() & KBD_MODE_KCC)) { |
| /* |
| * If the controller does not support conversion, |
| * Set the keyboard to scan-code set 1. |
| */ |
| kbd_write_output_w(0xF0); |
| kbd_wait_for_input(); |
| kbd_write_output_w(0x01); |
| kbd_wait_for_input(); |
| } |
| kbd_write_output_w(KBD_CMD_ENABLE); |
| if (kbd_wait_for_input() != KBD_REPLY_ACK) |
| return "Kbd: enable keyboard: no ACK"; |
| |
| /* |
| * Finally, set the typematic rate to maximum. |
| */ |
| kbd_write_output_w(KBD_CMD_SET_RATE); |
| if (kbd_wait_for_input() != KBD_REPLY_ACK) |
| return "Kbd: Set rate: no ACK"; |
| kbd_write_output_w(0x00); |
| if (kbd_wait_for_input() != KBD_REPLY_ACK) |
| return "Kbd: Set rate: no ACK"; |
| return NULL; |
| } |
| |
| static void kbd_interrupt(void *dev_id) |
| { |
| handle_kbd_event(); |
| } |
| |
| /****************************************************************** |
| * Init |
| ******************************************************************/ |
| |
| int kbd_init_hw(void) |
| { |
| char* result; |
| |
| kbd_request_region(); |
| |
| result=kbd_initialize(); |
| if (result==NULL) { |
| PRINTF("AT Keyboard initialized\n"); |
| kbd_request_irq(kbd_interrupt); |
| return (1); |
| } else { |
| printf("%s\n",result); |
| return (-1); |
| } |
| } |
| |
| void pckbd_leds(unsigned char leds) |
| { |
| kbd_send_data(KBD_CMD_SET_LEDS); |
| kbd_send_data(leds); |
| } |
| |
| #endif /* CONFIG_PS2KBD */ |