blob: 1606ab33ff413cfe351b9b2248ec2f69debaff56 [file] [log] [blame]
wdenkb983fa22004-01-16 00:30:56 +00001/***********************************************************************
2 *
3 * (C) Copyright 2004
4 * DENX Software Engineering
5 * Wolfgang Denk, wd@denx.de
wdenkb983fa22004-01-16 00:30:56 +00006 *
7 * PS/2 keyboard driver
8 *
9 * Originally from linux source (drivers/char/pc_keyb.c)
10 *
11 ***********************************************************************/
12
13#include <common.h>
14
wdenkb983fa22004-01-16 00:30:56 +000015#include <keyboard.h>
16#include <pc_keyb.h>
17
18#undef KBG_DEBUG
19
20#ifdef KBG_DEBUG
21#define PRINTF(fmt,args...) printf (fmt ,##args)
22#else
23#define PRINTF(fmt,args...)
24#endif
25
26
27/*
28 * This reads the keyboard status port, and does the
29 * appropriate action.
30 *
31 */
32static unsigned char handle_kbd_event(void)
33{
34 unsigned char status = kbd_read_status();
35 unsigned int work = 10000;
36
37 while ((--work > 0) && (status & KBD_STAT_OBF)) {
38 unsigned char scancode;
39
40 scancode = kbd_read_input();
41
42 /* Error bytes must be ignored to make the
43 Synaptics touchpads compaq use work */
44 /* Ignore error bytes */
45 if (!(status & (KBD_STAT_GTO | KBD_STAT_PERR))) {
46 if (status & KBD_STAT_MOUSE_OBF)
47 ; /* not supported: handle_mouse_event(scancode); */
48 else
49 handle_scancode(scancode);
50 }
51 status = kbd_read_status();
52 }
53 if (!work)
54 PRINTF("pc_keyb: controller jammed (0x%02X).\n", status);
55 return status;
56}
57
58
59static int kbd_read_data(void)
60{
61 int val;
62 unsigned char status;
63
Wolfgang Denkdc770c72008-07-14 15:19:07 +020064 val = -1;
wdenkb983fa22004-01-16 00:30:56 +000065 status = kbd_read_status();
66 if (status & KBD_STAT_OBF) {
67 val = kbd_read_input();
68 if (status & (KBD_STAT_GTO | KBD_STAT_PERR))
69 val = -2;
70 }
71 return val;
72}
73
74static int kbd_wait_for_input(void)
75{
76 unsigned long timeout;
77 int val;
78
79 timeout = KBD_TIMEOUT;
80 val=kbd_read_data();
81 while(val < 0) {
82 if(timeout--==0)
83 return -1;
84 udelay(1000);
85 val=kbd_read_data();
86 }
87 return val;
88}
89
90
91static int kb_wait(void)
92{
93 unsigned long timeout = KBC_TIMEOUT * 10;
94
95 do {
96 unsigned char status = handle_kbd_event();
97 if (!(status & KBD_STAT_IBF))
98 return 0; /* ok */
99 udelay(1000);
100 timeout--;
101 } while (timeout);
102 return 1;
103}
104
105static void kbd_write_command_w(int data)
106{
107 if(kb_wait())
108 PRINTF("timeout in kbd_write_command_w\n");
109 kbd_write_command(data);
110}
111
112static void kbd_write_output_w(int data)
113{
114 if(kb_wait())
115 PRINTF("timeout in kbd_write_output_w\n");
116 kbd_write_output(data);
117}
118
119static void kbd_send_data(unsigned char data)
120{
121 kbd_write_output_w(data);
122 kbd_wait_for_input();
123}
124
125
126static char * kbd_initialize(void)
127{
128 int status;
129
130 /*
131 * Test the keyboard interface.
132 * This seems to be the only way to get it going.
133 * If the test is successful a x55 is placed in the input buffer.
134 */
135 kbd_write_command_w(KBD_CCMD_SELF_TEST);
136 if (kbd_wait_for_input() != 0x55)
137 return "Kbd: failed self test";
138 /*
139 * Perform a keyboard interface test. This causes the controller
140 * to test the keyboard clock and data lines. The results of the
141 * test are placed in the input buffer.
142 */
143 kbd_write_command_w(KBD_CCMD_KBD_TEST);
144 if (kbd_wait_for_input() != 0x00)
145 return "Kbd: interface failed self test";
146 /*
147 * Enable the keyboard by allowing the keyboard clock to run.
148 */
149 kbd_write_command_w(KBD_CCMD_KBD_ENABLE);
150
151 /*
152 * Reset keyboard. If the read times out
153 * then the assumption is that no keyboard is
154 * plugged into the machine.
155 * This defaults the keyboard to scan-code set 2.
156 *
157 * Set up to try again if the keyboard asks for RESEND.
158 */
159 do {
160 kbd_write_output_w(KBD_CMD_RESET);
161 status = kbd_wait_for_input();
162 if (status == KBD_REPLY_ACK)
163 break;
164 if (status != KBD_REPLY_RESEND) {
165 PRINTF("status: %X\n",status);
166 return "Kbd: reset failed, no ACK";
167 }
168 } while (1);
169 if (kbd_wait_for_input() != KBD_REPLY_POR)
170 return "Kbd: reset failed, no POR";
171
172 /*
173 * Set keyboard controller mode. During this, the keyboard should be
174 * in the disabled state.
175 *
176 * Set up to try again if the keyboard asks for RESEND.
177 */
178 do {
179 kbd_write_output_w(KBD_CMD_DISABLE);
180 status = kbd_wait_for_input();
181 if (status == KBD_REPLY_ACK)
182 break;
183 if (status != KBD_REPLY_RESEND)
184 return "Kbd: disable keyboard: no ACK";
185 } while (1);
186
187 kbd_write_command_w(KBD_CCMD_WRITE_MODE);
188 kbd_write_output_w(KBD_MODE_KBD_INT
189 | KBD_MODE_SYS
190 | KBD_MODE_DISABLE_MOUSE
191 | KBD_MODE_KCC);
192
Wolfgang Denk0ee70772005-09-23 11:05:55 +0200193 /* AMCC powerpc portables need this to use scan-code set 1 -- Cort */
wdenkb983fa22004-01-16 00:30:56 +0000194 kbd_write_command_w(KBD_CCMD_READ_MODE);
195 if (!(kbd_wait_for_input() & KBD_MODE_KCC)) {
196 /*
197 * If the controller does not support conversion,
198 * Set the keyboard to scan-code set 1.
199 */
200 kbd_write_output_w(0xF0);
201 kbd_wait_for_input();
202 kbd_write_output_w(0x01);
203 kbd_wait_for_input();
204 }
205 kbd_write_output_w(KBD_CMD_ENABLE);
206 if (kbd_wait_for_input() != KBD_REPLY_ACK)
207 return "Kbd: enable keyboard: no ACK";
208
209 /*
210 * Finally, set the typematic rate to maximum.
211 */
212 kbd_write_output_w(KBD_CMD_SET_RATE);
213 if (kbd_wait_for_input() != KBD_REPLY_ACK)
214 return "Kbd: Set rate: no ACK";
215 kbd_write_output_w(0x00);
216 if (kbd_wait_for_input() != KBD_REPLY_ACK)
217 return "Kbd: Set rate: no ACK";
218 return NULL;
219}
220
221static void kbd_interrupt(void *dev_id)
222{
223 handle_kbd_event();
224}
225
226/******************************************************************
227 * Init
228 ******************************************************************/
229
230int kbd_init_hw(void)
231{
232 char* result;
233
234 kbd_request_region();
235
236 result=kbd_initialize();
237 if (result==NULL) {
238 PRINTF("AT Keyboard initialized\n");
239 kbd_request_irq(kbd_interrupt);
240 return (1);
241 } else {
242 printf("%s\n",result);
243 return (-1);
244 }
245}
246
247void pckbd_leds(unsigned char leds)
248{
249 kbd_send_data(KBD_CMD_SET_LEDS);
250 kbd_send_data(leds);
251}