3c41e5f60250194bdd1238efec536ed26c848465
[qemu] / hw / pckbd.c
1 /*
2  * QEMU PC keyboard emulation
3  * 
4  * Copyright (c) 2003 Fabrice Bellard
5  * 
6  * Permission is hereby granted, free of charge, to any person obtaining a copy
7  * of this software and associated documentation files (the "Software"), to deal
8  * in the Software without restriction, including without limitation the rights
9  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
10  * copies of the Software, and to permit persons to whom the Software is
11  * furnished to do so, subject to the following conditions:
12  *
13  * The above copyright notice and this permission notice shall be included in
14  * all copies or substantial portions of the Software.
15  *
16  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
19  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
22  * THE SOFTWARE.
23  */
24 #include "vl.h"
25
26 /* debug PC keyboard */
27 //#define DEBUG_KBD
28
29 /* debug PC keyboard : only mouse */
30 //#define DEBUG_MOUSE
31
32 /*      Keyboard Controller Commands */
33 #define KBD_CCMD_READ_MODE      0x20    /* Read mode bits */
34 #define KBD_CCMD_WRITE_MODE     0x60    /* Write mode bits */
35 #define KBD_CCMD_GET_VERSION    0xA1    /* Get controller version */
36 #define KBD_CCMD_MOUSE_DISABLE  0xA7    /* Disable mouse interface */
37 #define KBD_CCMD_MOUSE_ENABLE   0xA8    /* Enable mouse interface */
38 #define KBD_CCMD_TEST_MOUSE     0xA9    /* Mouse interface test */
39 #define KBD_CCMD_SELF_TEST      0xAA    /* Controller self test */
40 #define KBD_CCMD_KBD_TEST       0xAB    /* Keyboard interface test */
41 #define KBD_CCMD_KBD_DISABLE    0xAD    /* Keyboard interface disable */
42 #define KBD_CCMD_KBD_ENABLE     0xAE    /* Keyboard interface enable */
43 #define KBD_CCMD_READ_INPORT    0xC0    /* read input port */
44 #define KBD_CCMD_READ_OUTPORT   0xD0    /* read output port */
45 #define KBD_CCMD_WRITE_OUTPORT  0xD1    /* write output port */
46 #define KBD_CCMD_WRITE_OBUF     0xD2
47 #define KBD_CCMD_WRITE_AUX_OBUF 0xD3    /* Write to output buffer as if
48                                            initiated by the auxiliary device */
49 #define KBD_CCMD_WRITE_MOUSE    0xD4    /* Write the following byte to the mouse */
50 #define KBD_CCMD_DISABLE_A20    0xDD    /* HP vectra only ? */
51 #define KBD_CCMD_ENABLE_A20     0xDF    /* HP vectra only ? */
52 #define KBD_CCMD_RESET          0xFE
53
54 /* Keyboard Commands */
55 #define KBD_CMD_SET_LEDS        0xED    /* Set keyboard leds */
56 #define KBD_CMD_ECHO            0xEE
57 #define KBD_CMD_GET_ID          0xF2    /* get keyboard ID */
58 #define KBD_CMD_SET_RATE        0xF3    /* Set typematic rate */
59 #define KBD_CMD_ENABLE          0xF4    /* Enable scanning */
60 #define KBD_CMD_RESET_DISABLE   0xF5    /* reset and disable scanning */
61 #define KBD_CMD_RESET_ENABLE    0xF6    /* reset and enable scanning */
62 #define KBD_CMD_RESET           0xFF    /* Reset */
63
64 /* Keyboard Replies */
65 #define KBD_REPLY_POR           0xAA    /* Power on reset */
66 #define KBD_REPLY_ACK           0xFA    /* Command ACK */
67 #define KBD_REPLY_RESEND        0xFE    /* Command NACK, send the cmd again */
68
69 /* Status Register Bits */
70 #define KBD_STAT_OBF            0x01    /* Keyboard output buffer full */
71 #define KBD_STAT_IBF            0x02    /* Keyboard input buffer full */
72 #define KBD_STAT_SELFTEST       0x04    /* Self test successful */
73 #define KBD_STAT_CMD            0x08    /* Last write was a command write (0=data) */
74 #define KBD_STAT_UNLOCKED       0x10    /* Zero if keyboard locked */
75 #define KBD_STAT_MOUSE_OBF      0x20    /* Mouse output buffer full */
76 #define KBD_STAT_GTO            0x40    /* General receive/xmit timeout */
77 #define KBD_STAT_PERR           0x80    /* Parity error */
78
79 /* Controller Mode Register Bits */
80 #define KBD_MODE_KBD_INT        0x01    /* Keyboard data generate IRQ1 */
81 #define KBD_MODE_MOUSE_INT      0x02    /* Mouse data generate IRQ12 */
82 #define KBD_MODE_SYS            0x04    /* The system flag (?) */
83 #define KBD_MODE_NO_KEYLOCK     0x08    /* The keylock doesn't affect the keyboard if set */
84 #define KBD_MODE_DISABLE_KBD    0x10    /* Disable keyboard interface */
85 #define KBD_MODE_DISABLE_MOUSE  0x20    /* Disable mouse interface */
86 #define KBD_MODE_KCC            0x40    /* Scan code conversion to PC format */
87 #define KBD_MODE_RFU            0x80
88
89 /* Mouse Commands */
90 #define AUX_SET_SCALE11         0xE6    /* Set 1:1 scaling */
91 #define AUX_SET_SCALE21         0xE7    /* Set 2:1 scaling */
92 #define AUX_SET_RES             0xE8    /* Set resolution */
93 #define AUX_GET_SCALE           0xE9    /* Get scaling factor */
94 #define AUX_SET_STREAM          0xEA    /* Set stream mode */
95 #define AUX_POLL                0xEB    /* Poll */
96 #define AUX_RESET_WRAP          0xEC    /* Reset wrap mode */
97 #define AUX_SET_WRAP            0xEE    /* Set wrap mode */
98 #define AUX_SET_REMOTE          0xF0    /* Set remote mode */
99 #define AUX_GET_TYPE            0xF2    /* Get type */
100 #define AUX_SET_SAMPLE          0xF3    /* Set sample rate */
101 #define AUX_ENABLE_DEV          0xF4    /* Enable aux device */
102 #define AUX_DISABLE_DEV         0xF5    /* Disable aux device */
103 #define AUX_SET_DEFAULT         0xF6
104 #define AUX_RESET               0xFF    /* Reset aux device */
105 #define AUX_ACK                 0xFA    /* Command byte ACK. */
106
107 #define MOUSE_STATUS_REMOTE     0x40
108 #define MOUSE_STATUS_ENABLED    0x20
109 #define MOUSE_STATUS_SCALE21    0x10
110
111 #define KBD_QUEUE_SIZE 256
112
113 #define KBD_PENDING_KBD         1
114 #define KBD_PENDING_AUX         2
115
116 typedef struct KBDState {
117     uint8_t write_cmd; /* if non zero, write data to port 60 is expected */
118     uint8_t status;
119     uint8_t mode;
120     /* Bitmask of devices with data available.  */
121     uint8_t pending;
122     void *kbd;
123     void *mouse;
124 } KBDState;
125
126 KBDState kbd_state;
127
128 /* update irq and KBD_STAT_[MOUSE_]OBF */
129 /* XXX: not generating the irqs if KBD_MODE_DISABLE_KBD is set may be
130    incorrect, but it avoids having to simulate exact delays */
131 static void kbd_update_irq(KBDState *s)
132 {
133     int irq12_level, irq1_level;
134
135     irq1_level = 0;    
136     irq12_level = 0;    
137     s->status &= ~(KBD_STAT_OBF | KBD_STAT_MOUSE_OBF);
138     if (s->pending) {
139         s->status |= KBD_STAT_OBF;
140         /* kdb data takes priority over aux data.  */
141         if (s->pending == KBD_PENDING_AUX) {
142             s->status |= KBD_STAT_MOUSE_OBF;
143             if (s->mode & KBD_MODE_MOUSE_INT)
144                 irq12_level = 1;
145         } else {
146             if ((s->mode & KBD_MODE_KBD_INT) && 
147                 !(s->mode & KBD_MODE_DISABLE_KBD))
148                 irq1_level = 1;
149         }
150     }
151     pic_set_irq(1, irq1_level);
152     pic_set_irq(12, irq12_level);
153 }
154
155 static void kbd_update_kbd_irq(void *opaque, int level)
156 {
157     KBDState *s = (KBDState *)opaque;
158
159     if (level)
160         s->pending |= KBD_PENDING_KBD;
161     else
162         s->pending &= ~KBD_PENDING_KBD;
163     kbd_update_irq(s);
164 }
165
166 static void kbd_update_aux_irq(void *opaque, int level)
167 {
168     KBDState *s = (KBDState *)opaque;
169
170     if (level)
171         s->pending |= KBD_PENDING_AUX;
172     else
173         s->pending &= ~KBD_PENDING_AUX;
174     kbd_update_irq(s);
175 }
176
177 static uint32_t kbd_read_status(void *opaque, uint32_t addr)
178 {
179     KBDState *s = opaque;
180     int val;
181     val = s->status;
182 #if defined(DEBUG_KBD)
183     printf("kbd: read status=0x%02x\n", val);
184 #endif
185     return val;
186 }
187
188 static void kbd_queue(KBDState *s, int b, int aux)
189 {
190     if (aux)
191         ps2_queue(s->mouse, b);
192     else
193         ps2_queue(s->kbd, b);
194 }
195
196 static void kbd_write_command(void *opaque, uint32_t addr, uint32_t val)
197 {
198     KBDState *s = opaque;
199
200 #ifdef DEBUG_KBD
201     printf("kbd: write cmd=0x%02x\n", val);
202 #endif
203     switch(val) {
204     case KBD_CCMD_READ_MODE:
205         kbd_queue(s, s->mode, 0);
206         break;
207     case KBD_CCMD_WRITE_MODE:
208     case KBD_CCMD_WRITE_OBUF:
209     case KBD_CCMD_WRITE_AUX_OBUF:
210     case KBD_CCMD_WRITE_MOUSE:
211     case KBD_CCMD_WRITE_OUTPORT:
212         s->write_cmd = val;
213         break;
214     case KBD_CCMD_MOUSE_DISABLE:
215         s->mode |= KBD_MODE_DISABLE_MOUSE;
216         break;
217     case KBD_CCMD_MOUSE_ENABLE:
218         s->mode &= ~KBD_MODE_DISABLE_MOUSE;
219         break;
220     case KBD_CCMD_TEST_MOUSE:
221         kbd_queue(s, 0x00, 0);
222         break;
223     case KBD_CCMD_SELF_TEST:
224         s->status |= KBD_STAT_SELFTEST;
225         kbd_queue(s, 0x55, 0);
226         break;
227     case KBD_CCMD_KBD_TEST:
228         kbd_queue(s, 0x00, 0);
229         break;
230     case KBD_CCMD_KBD_DISABLE:
231         s->mode |= KBD_MODE_DISABLE_KBD;
232         kbd_update_irq(s);
233         break;
234     case KBD_CCMD_KBD_ENABLE:
235         s->mode &= ~KBD_MODE_DISABLE_KBD;
236         kbd_update_irq(s);
237         break;
238     case KBD_CCMD_READ_INPORT:
239         kbd_queue(s, 0x00, 0);
240         break;
241     case KBD_CCMD_READ_OUTPORT:
242         /* XXX: check that */
243 #ifdef TARGET_I386
244         val = 0x01 | (ioport_get_a20() << 1);
245 #else
246         val = 0x01;
247 #endif
248         if (s->status & KBD_STAT_OBF)
249             val |= 0x10;
250         if (s->status & KBD_STAT_MOUSE_OBF)
251             val |= 0x20;
252         kbd_queue(s, val, 0);
253         break;
254 #ifdef TARGET_I386
255     case KBD_CCMD_ENABLE_A20:
256         ioport_set_a20(1);
257         break;
258     case KBD_CCMD_DISABLE_A20:
259         ioport_set_a20(0);
260         break;
261 #endif
262     case KBD_CCMD_RESET:
263         qemu_system_reset_request();
264         break;
265     case 0xff:
266         /* ignore that - I don't know what is its use */
267         break;
268     default:
269         fprintf(stderr, "qemu: unsupported keyboard cmd=0x%02x\n", val);
270         break;
271     }
272 }
273
274 static uint32_t kbd_read_data(void *opaque, uint32_t addr)
275 {
276     KBDState *s = opaque;
277
278     if (s->pending == KBD_PENDING_AUX)
279         return ps2_read_data(s->mouse);
280
281     return ps2_read_data(s->kbd);
282 }
283
284 void kbd_write_data(void *opaque, uint32_t addr, uint32_t val)
285 {
286     KBDState *s = opaque;
287
288 #ifdef DEBUG_KBD
289     printf("kbd: write data=0x%02x\n", val);
290 #endif
291
292     switch(s->write_cmd) {
293     case 0:
294         ps2_write_keyboard(s->kbd, val);
295         break;
296     case KBD_CCMD_WRITE_MODE:
297         s->mode = val;
298         ps2_keyboard_set_translation(s->kbd, (s->mode & KBD_MODE_KCC) != 0);
299         /* ??? */
300         kbd_update_irq(s);
301         break;
302     case KBD_CCMD_WRITE_OBUF:
303         kbd_queue(s, val, 0);
304         break;
305     case KBD_CCMD_WRITE_AUX_OBUF:
306         kbd_queue(s, val, 1);
307         break;
308     case KBD_CCMD_WRITE_OUTPORT:
309 #ifdef TARGET_I386
310         ioport_set_a20((val >> 1) & 1);
311 #endif
312         if (!(val & 1)) {
313             qemu_system_reset_request();
314         }
315         break;
316     case KBD_CCMD_WRITE_MOUSE:
317         ps2_write_mouse(s->mouse, val);
318         break;
319     default:
320         break;
321     }
322     s->write_cmd = 0;
323 }
324
325 static void kbd_reset(void *opaque)
326 {
327     KBDState *s = opaque;
328
329     s->mode = KBD_MODE_KBD_INT | KBD_MODE_MOUSE_INT;
330     s->status = KBD_STAT_CMD | KBD_STAT_UNLOCKED;
331 }
332
333 static void kbd_save(QEMUFile* f, void* opaque)
334 {
335     KBDState *s = (KBDState*)opaque;
336     
337     qemu_put_8s(f, &s->write_cmd);
338     qemu_put_8s(f, &s->status);
339     qemu_put_8s(f, &s->mode);
340     qemu_put_8s(f, &s->pending);
341 }
342
343 static int kbd_load(QEMUFile* f, void* opaque, int version_id)
344 {
345     KBDState *s = (KBDState*)opaque;
346     
347     if (version_id != 3)
348         return -EINVAL;
349     qemu_get_8s(f, &s->write_cmd);
350     qemu_get_8s(f, &s->status);
351     qemu_get_8s(f, &s->mode);
352     qemu_get_8s(f, &s->pending);
353     return 0;
354 }
355
356 void kbd_init(void)
357 {
358     KBDState *s = &kbd_state;
359     
360     kbd_reset(s);
361     register_savevm("pckbd", 0, 3, kbd_save, kbd_load, s);
362     register_ioport_read(0x60, 1, 1, kbd_read_data, s);
363     register_ioport_write(0x60, 1, 1, kbd_write_data, s);
364     register_ioport_read(0x64, 1, 1, kbd_read_status, s);
365     register_ioport_write(0x64, 1, 1, kbd_write_command, s);
366
367     s->kbd = ps2_kbd_init(kbd_update_kbd_irq, s);
368     s->mouse = ps2_mouse_init(kbd_update_aux_irq, s);
369     qemu_register_reset(kbd_reset, s);
370 }