VirtualBox

source: vbox/trunk/src/VBox/Devices/Input/pckbd.c@ 788

Last change on this file since 788 was 481, checked in by vboxsync, 18 years ago

64-bit alignment.

  • Property svn:eol-style set to native
File size: 52.5 KB
Line 
1#ifdef VBOX
2/** @file
3 *
4 * VBox input devices:
5 * PS/2 keyboard & mouse controller device
6 */
7
8/*
9 * Copyright (C) 2006 InnoTek Systemberatung GmbH
10 *
11 * This file is part of VirtualBox Open Source Edition (OSE), as
12 * available from http://www.virtualbox.org. This file is free software;
13 * you can redistribute it and/or modify it under the terms of the GNU
14 * General Public License as published by the Free Software Foundation,
15 * in version 2 as it comes in the "COPYING" file of the VirtualBox OSE
16 * distribution. VirtualBox OSE is distributed in the hope that it will
17 * be useful, but WITHOUT ANY WARRANTY of any kind.
18 *
19 * If you received this file as part of a commercial VirtualBox
20 * distribution, then only the terms of your commercial VirtualBox
21 * license agreement apply instead of the previous paragraph.
22 *
23 * --------------------------------------------------------------------
24 *
25 * This code is based on:
26 *
27 * QEMU PC keyboard emulation (revision 1.12)
28 *
29 * Copyright (c) 2003 Fabrice Bellard
30 *
31 * Permission is hereby granted, free of charge, to any person obtaining a copy
32 * of this software and associated documentation files (the "Software"), to deal
33 * in the Software without restriction, including without limitation the rights
34 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
35 * copies of the Software, and to permit persons to whom the Software is
36 * furnished to do so, subject to the following conditions:
37 *
38 * The above copyright notice and this permission notice shall be included in
39 * all copies or substantial portions of the Software.
40 *
41 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
42 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
43 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
44 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
45 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
46 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
47 * THE SOFTWARE.
48 *
49 */
50
51
52/*******************************************************************************
53* Header Files *
54*******************************************************************************/
55#define LOG_GROUP LOG_GROUP_DEV_KBD
56#include "vl_vbox.h"
57#include <VBox/pdm.h>
58#include <VBox/err.h>
59
60#include <VBox/log.h>
61#include <iprt/assert.h>
62
63#include "Builtins.h"
64
65#define PCKBD_SAVED_STATE_VERSION 2
66
67
68#ifndef VBOX_DEVICE_STRUCT_TESTCASE
69/*******************************************************************************
70* Internal Functions *
71*******************************************************************************/
72PDMBOTHCBDECL(int) kbdIOPortDataRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb);
73PDMBOTHCBDECL(int) kbdIOPortDataWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb);
74PDMBOTHCBDECL(int) kbdIOPortStatusRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb);
75PDMBOTHCBDECL(int) kbdIOPortCommandWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb);
76
77#endif /* !VBOX_DEVICE_STRUCT_TESTCASE */
78#endif /* VBOX */
79
80
81#ifndef VBOX
82#include "vl.h"
83#endif
84
85#ifdef VBOX
86/* debug PC keyboard */
87#define DEBUG_KBD
88
89/* debug PC keyboard : only mouse */
90#define DEBUG_MOUSE
91#endif /* VBOX */
92
93/* Keyboard Controller Commands */
94#define KBD_CCMD_READ_MODE 0x20 /* Read mode bits */
95#define KBD_CCMD_WRITE_MODE 0x60 /* Write mode bits */
96#define KBD_CCMD_GET_VERSION 0xA1 /* Get controller version */
97#define KBD_CCMD_MOUSE_DISABLE 0xA7 /* Disable mouse interface */
98#define KBD_CCMD_MOUSE_ENABLE 0xA8 /* Enable mouse interface */
99#define KBD_CCMD_TEST_MOUSE 0xA9 /* Mouse interface test */
100#define KBD_CCMD_SELF_TEST 0xAA /* Controller self test */
101#define KBD_CCMD_KBD_TEST 0xAB /* Keyboard interface test */
102#define KBD_CCMD_KBD_DISABLE 0xAD /* Keyboard interface disable */
103#define KBD_CCMD_KBD_ENABLE 0xAE /* Keyboard interface enable */
104#define KBD_CCMD_READ_INPORT 0xC0 /* read input port */
105#define KBD_CCMD_READ_OUTPORT 0xD0 /* read output port */
106#define KBD_CCMD_WRITE_OUTPORT 0xD1 /* write output port */
107#define KBD_CCMD_WRITE_OBUF 0xD2
108#define KBD_CCMD_WRITE_AUX_OBUF 0xD3 /* Write to output buffer as if
109 initiated by the auxiliary device */
110#define KBD_CCMD_WRITE_MOUSE 0xD4 /* Write the following byte to the mouse */
111#define KBD_CCMD_DISABLE_A20 0xDD /* HP vectra only ? */
112#define KBD_CCMD_ENABLE_A20 0xDF /* HP vectra only ? */
113#define KBD_CCMD_RESET 0xFE
114
115/* Keyboard Commands */
116#define KBD_CMD_SET_LEDS 0xED /* Set keyboard leds */
117#define KBD_CMD_ECHO 0xEE
118#define KBD_CMD_GET_ID 0xF2 /* get keyboard ID */
119#define KBD_CMD_SET_RATE 0xF3 /* Set typematic rate */
120#define KBD_CMD_ENABLE 0xF4 /* Enable scanning */
121#define KBD_CMD_RESET_DISABLE 0xF5 /* reset and disable scanning */
122#define KBD_CMD_RESET_ENABLE 0xF6 /* reset and enable scanning */
123#define KBD_CMD_RESET 0xFF /* Reset */
124
125/* Keyboard Replies */
126#define KBD_REPLY_POR 0xAA /* Power on reset */
127#define KBD_REPLY_ACK 0xFA /* Command ACK */
128#define KBD_REPLY_RESEND 0xFE /* Command NACK, send the cmd again */
129
130/* Status Register Bits */
131#define KBD_STAT_OBF 0x01 /* Keyboard output buffer full */
132#define KBD_STAT_IBF 0x02 /* Keyboard input buffer full */
133#define KBD_STAT_SELFTEST 0x04 /* Self test successful */
134#define KBD_STAT_CMD 0x08 /* Last write was a command write (0=data) */
135#define KBD_STAT_UNLOCKED 0x10 /* Zero if keyboard locked */
136#define KBD_STAT_MOUSE_OBF 0x20 /* Mouse output buffer full */
137#define KBD_STAT_GTO 0x40 /* General receive/xmit timeout */
138#define KBD_STAT_PERR 0x80 /* Parity error */
139
140/* Controller Mode Register Bits */
141#define KBD_MODE_KBD_INT 0x01 /* Keyboard data generate IRQ1 */
142#define KBD_MODE_MOUSE_INT 0x02 /* Mouse data generate IRQ12 */
143#define KBD_MODE_SYS 0x04 /* The system flag (?) */
144#define KBD_MODE_NO_KEYLOCK 0x08 /* The keylock doesn't affect the keyboard if set */
145#define KBD_MODE_DISABLE_KBD 0x10 /* Disable keyboard interface */
146#define KBD_MODE_DISABLE_MOUSE 0x20 /* Disable mouse interface */
147#define KBD_MODE_KCC 0x40 /* Scan code conversion to PC format */
148#define KBD_MODE_RFU 0x80
149
150/* Mouse Commands */
151#define AUX_SET_SCALE11 0xE6 /* Set 1:1 scaling */
152#define AUX_SET_SCALE21 0xE7 /* Set 2:1 scaling */
153#define AUX_SET_RES 0xE8 /* Set resolution */
154#define AUX_GET_SCALE 0xE9 /* Get scaling factor */
155#define AUX_SET_STREAM 0xEA /* Set stream mode */
156#define AUX_POLL 0xEB /* Poll */
157#define AUX_RESET_WRAP 0xEC /* Reset wrap mode */
158#define AUX_SET_WRAP 0xEE /* Set wrap mode */
159#define AUX_SET_REMOTE 0xF0 /* Set remote mode */
160#define AUX_GET_TYPE 0xF2 /* Get type */
161#define AUX_SET_SAMPLE 0xF3 /* Set sample rate */
162#define AUX_ENABLE_DEV 0xF4 /* Enable aux device */
163#define AUX_DISABLE_DEV 0xF5 /* Disable aux device */
164#define AUX_SET_DEFAULT 0xF6
165#define AUX_RESET 0xFF /* Reset aux device */
166#define AUX_ACK 0xFA /* Command byte ACK. */
167
168#define MOUSE_STATUS_REMOTE 0x40
169#define MOUSE_STATUS_ENABLED 0x20
170#define MOUSE_STATUS_SCALE21 0x10
171
172#define KBD_QUEUE_SIZE 256
173
174typedef struct {
175#ifndef VBOX
176 uint8_t aux[KBD_QUEUE_SIZE];
177#endif /* !VBOX */
178 uint8_t data[KBD_QUEUE_SIZE];
179 int rptr, wptr, count;
180} KBDQueue;
181
182#ifdef VBOX
183
184#define MOUSE_CMD_QUEUE_SIZE 8
185
186typedef struct {
187 uint8_t data[MOUSE_CMD_QUEUE_SIZE];
188 int rptr, wptr, count;
189} MouseCmdQueue;
190
191
192#define MOUSE_EVENT_QUEUE_SIZE 256
193
194typedef struct {
195 uint8_t data[MOUSE_EVENT_QUEUE_SIZE];
196 int rptr, wptr, count;
197} MouseEventQueue;
198
199#endif /* VBOX */
200
201typedef struct KBDState {
202 KBDQueue queue;
203#ifdef VBOX
204 MouseCmdQueue mouse_command_queue;
205 MouseEventQueue mouse_event_queue;
206#endif /* VBOX */
207 uint8_t write_cmd; /* if non zero, write data to port 60 is expected */
208 uint8_t status;
209 uint8_t mode;
210 /* keyboard state */
211 int32_t kbd_write_cmd;
212 int32_t scan_enabled;
213 /* mouse state */
214 int32_t mouse_write_cmd;
215 uint8_t mouse_status;
216 uint8_t mouse_resolution;
217 uint8_t mouse_sample_rate;
218 uint8_t mouse_wrap;
219 uint8_t mouse_type; /* 0 = PS2, 3 = IMPS/2, 4 = IMEX */
220 uint8_t mouse_detect_state;
221 int32_t mouse_dx; /* current values, needed for 'poll' mode */
222 int32_t mouse_dy;
223 int32_t mouse_dz;
224 uint8_t mouse_buttons;
225
226#ifdef VBOX
227 /** Pointer to the device instance. */
228 PPDMDEVINSGC pDevInsGC;
229 /** Pointer to the device instance. */
230 PPDMDEVINSHC pDevInsHC;
231 /**
232 * Keyboard port - LUN#0.
233 */
234 struct
235 {
236 /** The base interface for the keyboard port. */
237 PDMIBASE Base;
238 /** The keyboard port base interface. */
239 PDMIKEYBOARDPORT Port;
240
241 /** The base interface of the attached keyboard driver. */
242 HCPTRTYPE(PPDMIBASE) pDrvBase;
243 /** The keyboard interface of the attached keyboard driver. */
244 HCPTRTYPE(PPDMIKEYBOARDCONNECTOR) pDrv;
245 } Keyboard;
246
247 /**
248 * Mouse port - LUN#1.
249 */
250 struct
251 {
252 /** The base interface for the mouse port. */
253 PDMIBASE Base;
254 /** The mouse port base interface. */
255 PDMIMOUSEPORT Port;
256
257 /** The base interface of the attached mouse driver. */
258 HCPTRTYPE(PPDMIBASE) pDrvBase;
259 /** The mouse interface of the attached mouse driver. */
260 HCPTRTYPE(PPDMIMOUSECONNECTOR) pDrv;
261 } Mouse;
262#endif
263} KBDState;
264
265#ifndef VBOX_DEVICE_STRUCT_TESTCASE
266#ifndef VBOX
267KBDState kbd_state;
268#endif
269
270/* update irq and KBD_STAT_[MOUSE_]OBF */
271/* XXX: not generating the irqs if KBD_MODE_DISABLE_KBD is set may be
272 incorrect, but it avoids having to simulate exact delays */
273static void kbd_update_irq(KBDState *s)
274{
275 KBDQueue *q = &s->queue;
276#ifdef VBOX
277 MouseCmdQueue *mcq = &s->mouse_command_queue;
278 MouseEventQueue *meq = &s->mouse_event_queue;
279#endif /* VBOX */
280 int irq12_level, irq1_level;
281
282 irq1_level = 0;
283 irq12_level = 0;
284 s->status &= ~(KBD_STAT_OBF | KBD_STAT_MOUSE_OBF);
285#ifdef VBOX
286 if (q->count != 0)
287 {
288 s->status |= KBD_STAT_OBF;
289 if ((s->mode & KBD_MODE_KBD_INT) && !(s->mode & KBD_MODE_DISABLE_KBD))
290 irq1_level = 1;
291 }
292 else if (mcq->count != 0 || meq->count != 0)
293 {
294 s->status |= KBD_STAT_OBF | KBD_STAT_MOUSE_OBF;
295 if (s->mode & KBD_MODE_MOUSE_INT)
296 irq12_level = 1;
297 }
298#else /* !VBOX */
299 if (q->count != 0) {
300 s->status |= KBD_STAT_OBF;
301 if (q->aux[q->rptr]) {
302 s->status |= KBD_STAT_MOUSE_OBF;
303 if (s->mode & KBD_MODE_MOUSE_INT)
304 irq12_level = 1;
305 } else {
306 if ((s->mode & KBD_MODE_KBD_INT) &&
307 !(s->mode & KBD_MODE_DISABLE_KBD))
308 irq1_level = 1;
309 }
310 }
311#endif /* !VBOX */
312#ifndef VBOX
313 pic_set_irq(1, irq1_level);
314 pic_set_irq(12, irq12_level);
315#else /* VBOX */
316 PDMDevHlpISASetIrq(CTXSUFF(s->pDevIns), 1, irq1_level);
317 PDMDevHlpISASetIrq(CTXSUFF(s->pDevIns), 12, irq12_level);
318#endif /* VBOX */
319}
320
321static void kbd_queue(KBDState *s, int b, int aux)
322{
323 KBDQueue *q = &s->queue;
324#ifdef VBOX
325 MouseCmdQueue *mcq = &s->mouse_command_queue;
326 MouseEventQueue *meq = &s->mouse_event_queue;
327#endif /* VBOX */
328
329#if defined(DEBUG_MOUSE) || defined(DEBUG_KBD)
330 if (aux)
331 Log(("mouse event: 0x%02x\n", b));
332#ifdef DEBUG_KBD
333 else
334 Log(("kbd event: 0x%02x\n", b));
335#endif
336#endif
337#ifdef VBOX
338 switch (aux)
339 {
340 case 0: /* keyboard */
341 if (q->count >= KBD_QUEUE_SIZE)
342 return;
343 q->data[q->wptr] = b;
344 if (++q->wptr == KBD_QUEUE_SIZE)
345 q->wptr = 0;
346 q->count++;
347 break;
348 case 1: /* mouse command response */
349 if (mcq->count >= MOUSE_CMD_QUEUE_SIZE)
350 return;
351 mcq->data[mcq->wptr] = b;
352 if (++mcq->wptr == MOUSE_CMD_QUEUE_SIZE)
353 mcq->wptr = 0;
354 mcq->count++;
355 break;
356 case 2: /* mouse event data */
357 if (meq->count >= MOUSE_EVENT_QUEUE_SIZE)
358 return;
359 meq->data[meq->wptr] = b;
360 if (++meq->wptr == MOUSE_EVENT_QUEUE_SIZE)
361 meq->wptr = 0;
362 meq->count++;
363 break;
364 default:
365 AssertMsgFailed(("aux=%d\n", aux));
366 }
367#else /* !VBOX */
368 if (q->count >= KBD_QUEUE_SIZE)
369 return;
370 q->aux[q->wptr] = aux;
371 q->data[q->wptr] = b;
372 if (++q->wptr == KBD_QUEUE_SIZE)
373 q->wptr = 0;
374 q->count++;
375#endif /* !VBOX */
376 kbd_update_irq(s);
377}
378
379#ifdef IN_RING3
380static void pc_kbd_put_keycode(void *opaque, int keycode)
381{
382 KBDState *s = opaque;
383 kbd_queue(s, keycode, 0);
384}
385#endif /* IN_RING3 */
386
387static uint32_t kbd_read_status(void *opaque, uint32_t addr)
388{
389 KBDState *s = opaque;
390 int val;
391 val = s->status;
392#if defined(DEBUG_KBD)
393 Log(("kbd: read status=0x%02x\n", val));
394#endif
395 return val;
396}
397
398#ifndef VBOX
399static void kbd_write_command(void *opaque, uint32_t addr, uint32_t val)
400{
401#else /* VBOX (we need VMReset return code passed back) */
402static int kbd_write_command(void *opaque, uint32_t addr, uint32_t val)
403{
404 int rc = VINF_SUCCESS;
405#endif /* VBOX */
406 KBDState *s = opaque;
407
408#ifdef DEBUG_KBD
409 Log(("kbd: write cmd=0x%02x\n", val));
410#endif
411 switch(val) {
412 case KBD_CCMD_READ_MODE:
413 kbd_queue(s, s->mode, 0);
414 break;
415 case KBD_CCMD_WRITE_MODE:
416 case KBD_CCMD_WRITE_OBUF:
417 case KBD_CCMD_WRITE_AUX_OBUF:
418 case KBD_CCMD_WRITE_MOUSE:
419 case KBD_CCMD_WRITE_OUTPORT:
420 s->write_cmd = val;
421 break;
422 case KBD_CCMD_MOUSE_DISABLE:
423 s->mode |= KBD_MODE_DISABLE_MOUSE;
424 break;
425 case KBD_CCMD_MOUSE_ENABLE:
426 s->mode &= ~KBD_MODE_DISABLE_MOUSE;
427 break;
428 case KBD_CCMD_TEST_MOUSE:
429 kbd_queue(s, 0x00, 0);
430 break;
431 case KBD_CCMD_SELF_TEST:
432 s->status |= KBD_STAT_SELFTEST;
433 kbd_queue(s, 0x55, 0);
434 break;
435 case KBD_CCMD_KBD_TEST:
436 kbd_queue(s, 0x00, 0);
437 break;
438 case KBD_CCMD_KBD_DISABLE:
439 s->mode |= KBD_MODE_DISABLE_KBD;
440 kbd_update_irq(s);
441 break;
442 case KBD_CCMD_KBD_ENABLE:
443 s->mode &= ~KBD_MODE_DISABLE_KBD;
444 kbd_update_irq(s);
445 break;
446 case KBD_CCMD_READ_INPORT:
447 kbd_queue(s, 0x00, 0);
448 break;
449 case KBD_CCMD_READ_OUTPORT:
450 /* XXX: check that */
451#ifdef TARGET_I386
452# ifndef VBOX
453 val = 0x01 | (ioport_get_a20() << 1);
454# else /* VBOX */
455 val = 0x01 | (PDMDevHlpA20IsEnabled(CTXSUFF(s->pDevIns)) << 1);
456# endif /* VBOX */
457#else
458 val = 0x01;
459#endif
460 if (s->status & KBD_STAT_OBF)
461 val |= 0x10;
462 if (s->status & KBD_STAT_MOUSE_OBF)
463 val |= 0x20;
464 kbd_queue(s, val, 0);
465 break;
466#ifdef TARGET_I386
467 case KBD_CCMD_ENABLE_A20:
468#ifndef VBOX
469 ioport_set_a20(1);
470#else /* VBOX */
471# ifndef IN_RING3
472 if (!PDMDevHlpA20IsEnabled(CTXSUFF(s->pDevIns)))
473 rc = VINF_IOM_HC_IOPORT_WRITE;
474# else /* IN_RING3 */
475 PDMDevHlpA20Set(CTXSUFF(s->pDevIns), true);
476# endif /* IN_RING3 */
477#endif /* VBOX */
478 break;
479 case KBD_CCMD_DISABLE_A20:
480#ifndef VBOX
481 ioport_set_a20(0);
482#else /* VBOX */
483# ifndef IN_RING3
484 if (PDMDevHlpA20IsEnabled(CTXSUFF(s->pDevIns)))
485 rc = VINF_IOM_HC_IOPORT_WRITE;
486# else /* IN_RING3 */
487 PDMDevHlpA20Set(CTXSUFF(s->pDevIns), false);
488# endif /* !IN_RING3 */
489#endif /* VBOX */
490 break;
491#endif
492 case KBD_CCMD_RESET:
493#ifndef VBOX
494 qemu_system_reset_request();
495#else /* VBOX */
496# ifndef IN_RING3
497 rc = VINF_IOM_HC_IOPORT_WRITE;
498# else /* IN_RING3 */
499 rc = PDMDevHlpVMReset(CTXSUFF(s->pDevIns));
500# endif /* !IN_RING3 */
501#endif /* VBOX */
502 break;
503 case 0xff:
504 /* ignore that - I don't know what is its use */
505 break;
506 default:
507 Log(("kbd: unsupported keyboard cmd=0x%02x\n", val));
508 break;
509 }
510#ifdef VBOX
511 return rc;
512#endif
513}
514
515static uint32_t kbd_read_data(void *opaque, uint32_t addr)
516{
517 KBDState *s = opaque;
518 KBDQueue *q;
519#ifdef VBOX
520 MouseCmdQueue *mcq;
521 MouseEventQueue *meq;
522#endif /* VBOX */
523 int val, index, aux;
524
525 q = &s->queue;
526#ifdef VBOX
527 mcq = &s->mouse_command_queue;
528 meq = &s->mouse_event_queue;
529 if (q->count == 0 && mcq->count == 0 && meq->count == 0) {
530#else /* !VBOX */
531 if (q->count == 0) {
532#endif /* !VBOX */
533 /* NOTE: if no data left, we return the last keyboard one
534 (needed for EMM386) */
535 /* XXX: need a timer to do things correctly */
536 index = q->rptr - 1;
537 if (index < 0)
538 index = KBD_QUEUE_SIZE - 1;
539 val = q->data[index];
540 } else {
541#ifdef VBOX
542 aux = (s->status & KBD_STAT_MOUSE_OBF);
543 if (!aux)
544 {
545 val = q->data[q->rptr];
546 if (++q->rptr == KBD_QUEUE_SIZE)
547 q->rptr = 0;
548 q->count--;
549 }
550 else
551 {
552 if (mcq->count)
553 {
554 val = mcq->data[mcq->rptr];
555 if (++mcq->rptr == MOUSE_CMD_QUEUE_SIZE)
556 mcq->rptr = 0;
557 mcq->count--;
558 }
559 else
560 {
561 val = meq->data[meq->rptr];
562 if (++meq->rptr == MOUSE_EVENT_QUEUE_SIZE)
563 meq->rptr = 0;
564 meq->count--;
565 }
566 }
567#else /* !VBOX */
568 aux = q->aux[q->rptr];
569 val = q->data[q->rptr];
570 if (++q->rptr == KBD_QUEUE_SIZE)
571 q->rptr = 0;
572 q->count--;
573#endif /* !VBOX */
574 /* reading deasserts IRQ */
575#ifndef VBOX
576 if (aux)
577 pic_set_irq(12, 0);
578 else
579 pic_set_irq(1, 0);
580#else /* VBOX */
581 if (aux)
582 PDMDevHlpISASetIrq(CTXSUFF(s->pDevIns), 12, 0);
583 else
584 PDMDevHlpISASetIrq(CTXSUFF(s->pDevIns), 1, 0);
585#endif /* VBOX */
586 }
587 /* reassert IRQs if data left */
588 kbd_update_irq(s);
589#ifdef DEBUG_KBD
590 Log(("kbd: read data=0x%02x\n", val));
591#endif
592 return val;
593}
594
595static void kbd_reset_keyboard(KBDState *s)
596{
597 s->scan_enabled = 1;
598}
599
600#ifndef VBOX
601static void kbd_write_keyboard(KBDState *s, int val)
602#else
603static int kbd_write_keyboard(KBDState *s, int val)
604#endif
605{
606 switch(s->kbd_write_cmd) {
607 default:
608 case -1:
609 switch(val) {
610 case 0x00:
611 kbd_queue(s, KBD_REPLY_ACK, 0);
612 break;
613 case 0x05:
614 kbd_queue(s, KBD_REPLY_RESEND, 0);
615 break;
616 case KBD_CMD_GET_ID:
617 kbd_queue(s, KBD_REPLY_ACK, 0);
618 kbd_queue(s, 0xab, 0);
619 kbd_queue(s, 0x83, 0);
620 break;
621 case KBD_CMD_ECHO:
622 kbd_queue(s, KBD_CMD_ECHO, 0);
623 break;
624 case KBD_CMD_ENABLE:
625 s->scan_enabled = 1;
626 kbd_queue(s, KBD_REPLY_ACK, 0);
627 break;
628 case KBD_CMD_SET_LEDS:
629 case KBD_CMD_SET_RATE:
630 s->kbd_write_cmd = val;
631 kbd_queue(s, KBD_REPLY_ACK, 0);
632 break;
633 case KBD_CMD_RESET_DISABLE:
634 kbd_reset_keyboard(s);
635 s->scan_enabled = 0;
636 kbd_queue(s, KBD_REPLY_ACK, 0);
637 break;
638 case KBD_CMD_RESET_ENABLE:
639 kbd_reset_keyboard(s);
640 s->scan_enabled = 1;
641 kbd_queue(s, KBD_REPLY_ACK, 0);
642 break;
643 case KBD_CMD_RESET:
644 kbd_reset_keyboard(s);
645 kbd_queue(s, KBD_REPLY_ACK, 0);
646 kbd_queue(s, KBD_REPLY_POR, 0);
647 break;
648 default:
649 kbd_queue(s, KBD_REPLY_ACK, 0);
650 break;
651 }
652 break;
653 case KBD_CMD_SET_LEDS:
654 {
655#ifdef IN_RING3
656 PDMKEYBLEDS enmLeds = 0
657 | ((val & 0x01) ? PDMKEYBLEDS_SCROLLLOCK : 0)
658 | ((val & 0x02) ? PDMKEYBLEDS_NUMLOCK : 0)
659 | ((val & 0x04) ? PDMKEYBLEDS_CAPSLOCK : 0)
660 ;
661 s->Keyboard.pDrv->pfnLedStatusChange(s->Keyboard.pDrv, enmLeds);
662#else
663 return VINF_IOM_HC_IOPORT_WRITE;
664#endif
665 kbd_queue(s, KBD_REPLY_ACK, 0);
666 s->kbd_write_cmd = -1;
667 }
668 break;
669 case KBD_CMD_SET_RATE:
670 kbd_queue(s, KBD_REPLY_ACK, 0);
671 s->kbd_write_cmd = -1;
672 break;
673 }
674
675 return VINF_SUCCESS;
676}
677
678#ifdef VBOX
679static void kbd_mouse_send_packet(KBDState *s, bool fToCmdQueue)
680#else /* !VBOX */
681static void kbd_mouse_send_packet(KBDState *s)
682#endif /* !VBOX */
683{
684#ifdef VBOX
685 int aux = fToCmdQueue ? 1 : 2;
686#endif /* VBOX */
687 unsigned int b;
688 int dx1, dy1, dz1;
689
690 dx1 = s->mouse_dx;
691 dy1 = s->mouse_dy;
692 dz1 = s->mouse_dz;
693 /* XXX: increase range to 8 bits ? */
694 if (dx1 > 127)
695 dx1 = 127;
696 else if (dx1 < -127)
697 dx1 = -127;
698 if (dy1 > 127)
699 dy1 = 127;
700 else if (dy1 < -127)
701 dy1 = -127;
702 b = 0x08 | ((dx1 < 0) << 4) | ((dy1 < 0) << 5) | (s->mouse_buttons & 0x07);
703#ifdef VBOX
704 kbd_queue(s, b, aux);
705 kbd_queue(s, dx1 & 0xff, aux);
706 kbd_queue(s, dy1 & 0xff, aux);
707#else /* !VBOX */
708 kbd_queue(s, b, 1);
709 kbd_queue(s, dx1 & 0xff, 1);
710 kbd_queue(s, dy1 & 0xff, 1);
711#endif /* !VBOX */
712 /* extra byte for IMPS/2 or IMEX */
713 switch(s->mouse_type) {
714 default:
715 break;
716 case 3:
717 if (dz1 > 127)
718 dz1 = 127;
719 else if (dz1 < -127)
720 dz1 = -127;
721#ifdef VBOX
722 kbd_queue(s, dz1 & 0xff, aux);
723#else /* !VBOX */
724 kbd_queue(s, dz1 & 0xff, 1);
725#endif /* !VBOX */
726 break;
727 case 4:
728 if (dz1 > 7)
729 dz1 = 7;
730 else if (dz1 < -7)
731 dz1 = -7;
732 b = (dz1 & 0x0f) | ((s->mouse_buttons & 0x18) << 1);
733#ifdef VBOX
734 kbd_queue(s, b, aux);
735#else /* !VBOX */
736 kbd_queue(s, b, 1);
737#endif /* !VBOX */
738 break;
739 }
740
741 /* update deltas */
742 s->mouse_dx -= dx1;
743 s->mouse_dy -= dy1;
744 s->mouse_dz -= dz1;
745}
746
747#ifdef IN_RING3
748static void pc_kbd_mouse_event(void *opaque,
749 int dx, int dy, int dz, int buttons_state)
750{
751 KBDState *s = opaque;
752
753 /* check if deltas are recorded when disabled */
754 if (!(s->mouse_status & MOUSE_STATUS_ENABLED))
755 return;
756
757 s->mouse_dx += dx;
758 s->mouse_dy -= dy;
759 s->mouse_dz += dz;
760 /* XXX: SDL sometimes generates nul events: we delete them */
761 if (s->mouse_dx == 0 && s->mouse_dy == 0 && s->mouse_dz == 0 &&
762 s->mouse_buttons == buttons_state)
763 return;
764 s->mouse_buttons = buttons_state;
765
766#ifdef VBOX
767 if (!(s->mouse_status & MOUSE_STATUS_REMOTE) &&
768 (s->mouse_event_queue.count < (MOUSE_EVENT_QUEUE_SIZE - 4))) {
769 for(;;) {
770 /* if not remote, send event. Multiple events are sent if
771 too big deltas */
772 kbd_mouse_send_packet(s, false);
773 if (s->mouse_dx == 0 && s->mouse_dy == 0 && s->mouse_dz == 0)
774 break;
775 }
776 }
777#else /* !VBOX */
778 if (!(s->mouse_status & MOUSE_STATUS_REMOTE) &&
779 (s->queue.count < (KBD_QUEUE_SIZE - 16))) {
780 for(;;) {
781 /* if not remote, send event. Multiple events are sent if
782 too big deltas */
783 kbd_mouse_send_packet(s);
784 if (s->mouse_dx == 0 && s->mouse_dy == 0 && s->mouse_dz == 0)
785 break;
786 }
787 }
788#endif /* !VBOX */
789}
790#endif /* IN_RING3 */
791
792static void kbd_write_mouse(KBDState *s, int val)
793{
794#ifdef DEBUG_MOUSE
795 Log(("kbd: write mouse 0x%02x\n", val));
796#endif
797#ifdef VBOX
798 /* Flush the mouse command response queue. */
799 s->mouse_command_queue.count = 0;
800 s->mouse_command_queue.rptr = 0;
801 s->mouse_command_queue.wptr = 0;
802#endif /* VBOX */
803 switch(s->mouse_write_cmd) {
804 default:
805 case -1:
806 /* mouse command */
807 if (s->mouse_wrap) {
808 if (val == AUX_RESET_WRAP) {
809 s->mouse_wrap = 0;
810 kbd_queue(s, AUX_ACK, 1);
811 return;
812 } else if (val != AUX_RESET) {
813 kbd_queue(s, val, 1);
814 return;
815 }
816 }
817 switch(val) {
818 case AUX_SET_SCALE11:
819 s->mouse_status &= ~MOUSE_STATUS_SCALE21;
820 kbd_queue(s, AUX_ACK, 1);
821 break;
822 case AUX_SET_SCALE21:
823 s->mouse_status |= MOUSE_STATUS_SCALE21;
824 kbd_queue(s, AUX_ACK, 1);
825 break;
826 case AUX_SET_STREAM:
827 s->mouse_status &= ~MOUSE_STATUS_REMOTE;
828 kbd_queue(s, AUX_ACK, 1);
829 break;
830 case AUX_SET_WRAP:
831 s->mouse_wrap = 1;
832 kbd_queue(s, AUX_ACK, 1);
833 break;
834 case AUX_SET_REMOTE:
835 s->mouse_status |= MOUSE_STATUS_REMOTE;
836 kbd_queue(s, AUX_ACK, 1);
837 break;
838 case AUX_GET_TYPE:
839 kbd_queue(s, AUX_ACK, 1);
840 kbd_queue(s, s->mouse_type, 1);
841 break;
842 case AUX_SET_RES:
843 case AUX_SET_SAMPLE:
844 s->mouse_write_cmd = val;
845 kbd_queue(s, AUX_ACK, 1);
846 break;
847 case AUX_GET_SCALE:
848 kbd_queue(s, AUX_ACK, 1);
849 kbd_queue(s, s->mouse_status, 1);
850 kbd_queue(s, s->mouse_resolution, 1);
851 kbd_queue(s, s->mouse_sample_rate, 1);
852 break;
853 case AUX_POLL:
854 kbd_queue(s, AUX_ACK, 1);
855#ifdef VBOX
856 kbd_mouse_send_packet(s, true);
857#else /* !VBOX */
858 kbd_mouse_send_packet(s);
859#endif /* !VBOX */
860 break;
861 case AUX_ENABLE_DEV:
862 s->mouse_status |= MOUSE_STATUS_ENABLED;
863 kbd_queue(s, AUX_ACK, 1);
864 break;
865 case AUX_DISABLE_DEV:
866 s->mouse_status &= ~MOUSE_STATUS_ENABLED;
867 kbd_queue(s, AUX_ACK, 1);
868#ifdef VBOX
869 /* Flush the mouse events queue. */
870 s->mouse_event_queue.count = 0;
871 s->mouse_event_queue.rptr = 0;
872 s->mouse_event_queue.wptr = 0;
873#endif /* VBOX */
874 break;
875 case AUX_SET_DEFAULT:
876 s->mouse_sample_rate = 100;
877 s->mouse_resolution = 2;
878 s->mouse_status = 0;
879 kbd_queue(s, AUX_ACK, 1);
880 break;
881 case AUX_RESET:
882 s->mouse_sample_rate = 100;
883 s->mouse_resolution = 2;
884 s->mouse_status = 0;
885 s->mouse_type = 0;
886 kbd_queue(s, AUX_ACK, 1);
887 kbd_queue(s, 0xaa, 1);
888 kbd_queue(s, s->mouse_type, 1);
889#ifdef VBOX
890 /* Flush the mouse events queue. */
891 s->mouse_event_queue.count = 0;
892 s->mouse_event_queue.rptr = 0;
893 s->mouse_event_queue.wptr = 0;
894#endif /* VBOX */
895 break;
896 default:
897 break;
898 }
899 break;
900 case AUX_SET_SAMPLE:
901 s->mouse_sample_rate = val;
902 /* detect IMPS/2 or IMEX */
903 switch(s->mouse_detect_state) {
904 default:
905 case 0:
906 if (val == 200)
907 s->mouse_detect_state = 1;
908 break;
909 case 1:
910 if (val == 100)
911 s->mouse_detect_state = 2;
912 else if (val == 200)
913 s->mouse_detect_state = 3;
914 else
915 s->mouse_detect_state = 0;
916 break;
917 case 2:
918 if (val == 80)
919 s->mouse_type = 3; /* IMPS/2 */
920 s->mouse_detect_state = 0;
921 break;
922 case 3:
923 if (val == 80)
924 s->mouse_type = 4; /* IMEX */
925 s->mouse_detect_state = 0;
926 break;
927 }
928 kbd_queue(s, AUX_ACK, 1);
929 s->mouse_write_cmd = -1;
930 break;
931 case AUX_SET_RES:
932 s->mouse_resolution = val;
933 kbd_queue(s, AUX_ACK, 1);
934 s->mouse_write_cmd = -1;
935 break;
936 }
937}
938
939#ifndef VBOX
940static void kbd_write_data(void *opaque, uint32_t addr, uint32_t val)
941{
942#else /* VBOX */
943static int kbd_write_data(void *opaque, uint32_t addr, uint32_t val)
944{
945 int rc = VINF_SUCCESS;
946#endif /* VBOX */
947 KBDState *s = opaque;
948
949#ifdef DEBUG_KBD
950 Log(("kbd: write data=0x%02x\n", val));
951#endif
952
953 switch(s->write_cmd) {
954 case 0:
955 rc = kbd_write_keyboard(s, val);
956 break;
957 case KBD_CCMD_WRITE_MODE:
958 s->mode = val;
959 kbd_update_irq(s);
960 break;
961 case KBD_CCMD_WRITE_OBUF:
962 kbd_queue(s, val, 0);
963 break;
964 case KBD_CCMD_WRITE_AUX_OBUF:
965 kbd_queue(s, val, 1);
966 break;
967 case KBD_CCMD_WRITE_OUTPORT:
968#ifdef TARGET_I386
969# ifndef VBOX
970 ioport_set_a20((val >> 1) & 1);
971# else /* VBOX */
972# ifndef IN_RING3
973 if (PDMDevHlpA20IsEnabled(CTXSUFF(s->pDevIns)) != !!(val & 2))
974 rc = VINF_IOM_HC_IOPORT_WRITE;
975# else /* IN_RING3 */
976 PDMDevHlpA20Set(CTXSUFF(s->pDevIns), !!(val & 2));
977# endif /* !IN_RING3 */
978# endif /* VBOX */
979#endif
980 if (!(val & 1)) {
981#ifndef VBOX
982 qemu_system_reset_request();
983#else /* VBOX */
984# ifndef IN_RING3
985 rc = VINF_IOM_HC_IOPORT_WRITE;
986# else
987 rc = PDMDevHlpVMReset(CTXSUFF(s->pDevIns));
988# endif
989#endif /* VBOX */
990 }
991 break;
992 case KBD_CCMD_WRITE_MOUSE:
993 kbd_write_mouse(s, val);
994 break;
995 default:
996 break;
997 }
998 s->write_cmd = 0;
999
1000#ifdef VBOX
1001 return rc;
1002#endif
1003}
1004
1005#ifdef IN_RING3
1006
1007static void kbd_reset(void *opaque)
1008{
1009 KBDState *s = opaque;
1010 KBDQueue *q;
1011#ifdef VBOX
1012 MouseCmdQueue *mcq;
1013 MouseEventQueue *meq;
1014#endif /* VBOX */
1015
1016 s->kbd_write_cmd = -1;
1017 s->mouse_write_cmd = -1;
1018 s->mode = KBD_MODE_KBD_INT | KBD_MODE_MOUSE_INT;
1019 s->status = KBD_STAT_CMD | KBD_STAT_UNLOCKED;
1020#ifdef VBOX /* Resetting everything, keyword was not working right on NT4 reboot. */
1021 s->write_cmd = 0;
1022 s->scan_enabled = 0;
1023 s->mouse_status = 0;
1024 s->mouse_resolution = 0;
1025 s->mouse_sample_rate = 0;
1026 s->mouse_wrap = 0;
1027 s->mouse_type = 0;
1028 s->mouse_detect_state = 0;
1029 s->mouse_dx = 0;
1030 s->mouse_dy = 0;
1031 s->mouse_dz = 0;
1032 s->mouse_buttons = 0;
1033#endif
1034 q = &s->queue;
1035 q->rptr = 0;
1036 q->wptr = 0;
1037 q->count = 0;
1038#ifdef VBOX
1039 mcq = &s->mouse_command_queue;
1040 mcq->rptr = 0;
1041 mcq->wptr = 0;
1042 mcq->count = 0;
1043 meq = &s->mouse_event_queue;
1044 meq->rptr = 0;
1045 meq->wptr = 0;
1046 meq->count = 0;
1047#endif /* VBOX */
1048}
1049
1050static void kbd_save(QEMUFile* f, void* opaque)
1051{
1052#ifdef VBOX
1053 uint32_t cItems;
1054 int i;
1055#endif /* VBOX */
1056 KBDState *s = (KBDState*)opaque;
1057
1058 qemu_put_8s(f, &s->write_cmd);
1059 qemu_put_8s(f, &s->status);
1060 qemu_put_8s(f, &s->mode);
1061 qemu_put_be32s(f, &s->kbd_write_cmd);
1062 qemu_put_be32s(f, &s->scan_enabled);
1063 qemu_put_be32s(f, &s->mouse_write_cmd);
1064 qemu_put_8s(f, &s->mouse_status);
1065 qemu_put_8s(f, &s->mouse_resolution);
1066 qemu_put_8s(f, &s->mouse_sample_rate);
1067 qemu_put_8s(f, &s->mouse_wrap);
1068 qemu_put_8s(f, &s->mouse_type);
1069 qemu_put_8s(f, &s->mouse_detect_state);
1070 qemu_put_be32s(f, &s->mouse_dx);
1071 qemu_put_be32s(f, &s->mouse_dy);
1072 qemu_put_be32s(f, &s->mouse_dz);
1073 qemu_put_8s(f, &s->mouse_buttons);
1074
1075#ifdef VBOX
1076 /*
1077 * We have to save the queues too.
1078 */
1079 cItems = s->queue.count;
1080 SSMR3PutU32(f, cItems);
1081 for (i = s->queue.rptr; cItems-- > 0; i = (i + 1) % ELEMENTS(s->queue.data))
1082 SSMR3PutU8(f, s->queue.data[i]);
1083 Log(("kbd_save: %d keyboard queue items stored\n", s->queue.count));
1084
1085 cItems = s->mouse_command_queue.count;
1086 SSMR3PutU32(f, cItems);
1087 for (i = s->mouse_command_queue.rptr; cItems-- > 0; i = (i + 1) % ELEMENTS(s->mouse_command_queue.data))
1088 SSMR3PutU8(f, s->mouse_command_queue.data[i]);
1089 Log(("kbd_save: %d mouse command queue items stored\n", s->mouse_command_queue.count));
1090
1091 cItems = s->mouse_event_queue.count;
1092 SSMR3PutU32(f, cItems);
1093 for (i = s->mouse_event_queue.rptr; cItems-- > 0; i = (i + 1) % ELEMENTS(s->mouse_event_queue.data))
1094 SSMR3PutU8(f, s->mouse_event_queue.data[i]);
1095 Log(("kbd_save: %d mouse event queue items stored\n", s->mouse_event_queue.count));
1096
1097 /* terminator */
1098 SSMR3PutU32(f, ~0);
1099#endif /* VBOX */
1100}
1101
1102static int kbd_load(QEMUFile* f, void* opaque, int version_id)
1103{
1104#ifdef VBOX
1105 uint32_t u32, i;
1106 int rc;
1107#endif
1108 KBDState *s = (KBDState*)opaque;
1109
1110 if (version_id != PCKBD_SAVED_STATE_VERSION)
1111#ifndef VBOX
1112 return -EINVAL;
1113#else
1114 return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
1115#endif
1116 qemu_get_8s(f, &s->write_cmd);
1117 qemu_get_8s(f, &s->status);
1118 qemu_get_8s(f, &s->mode);
1119 qemu_get_be32s(f, (uint32_t *)&s->kbd_write_cmd);
1120 qemu_get_be32s(f, (uint32_t *)&s->scan_enabled);
1121 qemu_get_be32s(f, (uint32_t *)&s->mouse_write_cmd);
1122 qemu_get_8s(f, &s->mouse_status);
1123 qemu_get_8s(f, &s->mouse_resolution);
1124 qemu_get_8s(f, &s->mouse_sample_rate);
1125 qemu_get_8s(f, &s->mouse_wrap);
1126 qemu_get_8s(f, &s->mouse_type);
1127 qemu_get_8s(f, &s->mouse_detect_state);
1128 qemu_get_be32s(f, (uint32_t *)&s->mouse_dx);
1129 qemu_get_be32s(f, (uint32_t *)&s->mouse_dy);
1130 qemu_get_be32s(f, (uint32_t *)&s->mouse_dz);
1131 qemu_get_8s(f, &s->mouse_buttons);
1132#ifdef VBOX
1133 s->queue.count = 0;
1134 s->queue.rptr = 0;
1135 s->queue.wptr = 0;
1136 s->mouse_command_queue.count = 0;
1137 s->mouse_command_queue.rptr = 0;
1138 s->mouse_command_queue.wptr = 0;
1139 s->mouse_event_queue.count = 0;
1140 s->mouse_event_queue.rptr = 0;
1141 s->mouse_event_queue.wptr = 0;
1142
1143 /*
1144 * Load the queues
1145 */
1146 rc = SSMR3GetU32(f, &u32);
1147 if (VBOX_FAILURE(rc))
1148 return rc;
1149 if (u32 > ELEMENTS(s->queue.data))
1150 {
1151 AssertMsgFailed(("u32=%#x\n", u32));
1152 return VERR_SSM_DATA_UNIT_FORMAT_CHANGED;
1153 }
1154 for (i = 0; i < u32; i++)
1155 {
1156 rc = SSMR3GetU8(f, &s->queue.data[i]);
1157 if (VBOX_FAILURE(rc))
1158 return rc;
1159 }
1160 s->queue.wptr = u32 % ELEMENTS(s->queue.data);
1161 s->queue.count = u32;
1162 Log(("kbd_load: %d keyboard queue items loaded\n", u32));
1163
1164 rc = SSMR3GetU32(f, &u32);
1165 if (VBOX_FAILURE(rc))
1166 return rc;
1167 if (u32 > ELEMENTS(s->mouse_command_queue.data))
1168 {
1169 AssertMsgFailed(("u32=%#x\n", u32));
1170 return VERR_SSM_DATA_UNIT_FORMAT_CHANGED;
1171 }
1172 for (i = 0; i < u32; i++)
1173 {
1174 rc = SSMR3GetU8(f, &s->mouse_command_queue.data[i]);
1175 if (VBOX_FAILURE(rc))
1176 return rc;
1177 }
1178 s->mouse_command_queue.wptr = u32 % ELEMENTS(s->mouse_command_queue.data);
1179 s->mouse_command_queue.count = u32;
1180 Log(("kbd_load: %d mouse command queue items loaded\n", u32));
1181
1182 rc = SSMR3GetU32(f, &u32);
1183 if (VBOX_FAILURE(rc))
1184 return rc;
1185 if (u32 > ELEMENTS(s->mouse_event_queue.data))
1186 {
1187 AssertMsgFailed(("u32=%#x\n", u32));
1188 return VERR_SSM_DATA_UNIT_FORMAT_CHANGED;
1189 }
1190 for (i = 0; i < u32; i++)
1191 {
1192 rc = SSMR3GetU8(f, &s->mouse_event_queue.data[i]);
1193 if (VBOX_FAILURE(rc))
1194 return rc;
1195 }
1196 s->mouse_event_queue.wptr = u32 % ELEMENTS(s->mouse_event_queue.data);
1197 s->mouse_event_queue.count = u32;
1198 Log(("kbd_load: %d mouse event queue items loaded\n", u32));
1199
1200 /* terminator */
1201 rc = SSMR3GetU32(f, &u32);
1202 if (VBOX_FAILURE(rc))
1203 return rc;
1204 if (u32 != ~0)
1205 {
1206 AssertMsgFailed(("u32=%#x\n", u32));
1207 return VERR_SSM_DATA_UNIT_FORMAT_CHANGED;
1208 }
1209#endif /* VBOX */
1210 return 0;
1211}
1212
1213#ifndef VBOX
1214void kbd_init(void)
1215{
1216 KBDState *s = &kbd_state;
1217
1218 kbd_reset(s);
1219 register_savevm("pckbd", 0, 1, kbd_save, kbd_load, s);
1220 register_ioport_read(0x60, 1, 1, kbd_read_data, s);
1221 register_ioport_write(0x60, 1, 1, kbd_write_data, s);
1222 register_ioport_read(0x64, 1, 1, kbd_read_status, s);
1223 register_ioport_write(0x64, 1, 1, kbd_write_command, s);
1224
1225 qemu_add_kbd_event_handler(pc_kbd_put_keycode, s);
1226 qemu_add_mouse_event_handler(pc_kbd_mouse_event, s);
1227 qemu_register_reset(kbd_reset, s);
1228}
1229#endif /* !VBOX */
1230
1231#endif /* IN_RING3 */
1232
1233
1234#ifdef VBOX /* InnoTek code start */
1235
1236/* -=-=-=-=-=- wrappers -=-=-=-=-=- */
1237
1238/**
1239 * Port I/O Handler for keyboard data IN operations.
1240 *
1241 * @returns VBox status code.
1242 *
1243 * @param pDevIns The device instance.
1244 * @param pvUser User argument - ignored.
1245 * @param Port Port number used for the IN operation.
1246 * @param pu32 Where to store the result.
1247 * @param cb Number of bytes read.
1248 */
1249PDMBOTHCBDECL(int) kbdIOPortDataRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb)
1250{
1251 NOREF(pvUser);
1252 if (cb == 1)
1253 {
1254 *pu32 = kbd_read_data(PDMINS2DATA(pDevIns, KBDState *), Port);
1255 Log2(("kbdIOPortDataRead: Port=%#x cb=%d *pu32=%#x\n", Port, cb, *pu32));
1256 return VINF_SUCCESS;
1257 }
1258 AssertMsgFailed(("Port=%#x cb=%d\n", Port, cb));
1259 return VERR_IOM_IOPORT_UNUSED;
1260}
1261
1262/**
1263 * Port I/O Handler for keyboard data OUT operations.
1264 *
1265 * @returns VBox status code.
1266 *
1267 * @param pDevIns The device instance.
1268 * @param pvUser User argument - ignored.
1269 * @param Port Port number used for the IN operation.
1270 * @param u32 The value to output.
1271 * @param cb The value size in bytes.
1272 */
1273PDMBOTHCBDECL(int) kbdIOPortDataWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
1274{
1275 int rc = VINF_SUCCESS;
1276 NOREF(pvUser);
1277 if (cb == 1)
1278 {
1279 rc = kbd_write_data(PDMINS2DATA(pDevIns, KBDState *), Port, u32);
1280 Log2(("kbdIOPortDataWrite: Port=%#x cb=%d u32=%#x\n", Port, cb, u32));
1281 }
1282 else
1283 AssertMsgFailed(("Port=%#x cb=%d\n", Port, cb));
1284 return rc;
1285}
1286
1287/**
1288 * Port I/O Handler for keyboard status IN operations.
1289 *
1290 * @returns VBox status code.
1291 *
1292 * @param pDevIns The device instance.
1293 * @param pvUser User argument - ignored.
1294 * @param Port Port number used for the IN operation.
1295 * @param pu32 Where to store the result.
1296 * @param cb Number of bytes read.
1297 */
1298PDMBOTHCBDECL(int) kbdIOPortStatusRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb)
1299{
1300 NOREF(pvUser);
1301 if (cb == 1)
1302 {
1303 *pu32 = kbd_read_status(PDMINS2DATA(pDevIns, KBDState *), Port);
1304 Log2(("kbdIOPortStatusRead: Port=%#x cb=%d -> *pu32=%#x\n", Port, cb, *pu32));
1305 return VINF_SUCCESS;
1306 }
1307 AssertMsgFailed(("Port=%#x cb=%d\n", Port, cb));
1308 return VERR_IOM_IOPORT_UNUSED;
1309}
1310
1311/**
1312 * Port I/O Handler for keyboard command OUT operations.
1313 *
1314 * @returns VBox status code.
1315 *
1316 * @param pDevIns The device instance.
1317 * @param pvUser User argument - ignored.
1318 * @param Port Port number used for the IN operation.
1319 * @param u32 The value to output.
1320 * @param cb The value size in bytes.
1321 */
1322PDMBOTHCBDECL(int) kbdIOPortCommandWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb)
1323{
1324 NOREF(pvUser);
1325 if (cb == 1)
1326 {
1327 int rc = kbd_write_command(PDMINS2DATA(pDevIns, KBDState *), Port, u32);
1328 Log2(("kbdIOPortCommandWrite: Port=%#x cb=%d u32=%#x rc=%Vrc\n", Port, cb, u32, rc));
1329 return rc;
1330 }
1331 AssertMsgFailed(("Port=%#x cb=%d\n", Port, cb));
1332 return VINF_SUCCESS;
1333}
1334
1335#ifdef IN_RING3
1336
1337/**
1338 * Saves a state of the keyboard device.
1339 *
1340 * @returns VBox status code.
1341 * @param pDevIns The device instance.
1342 * @param pSSMHandle The handle to save the state to.
1343 */
1344static DECLCALLBACK(int) kbdSaveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSMHandle)
1345{
1346 kbd_save(pSSMHandle, PDMINS2DATA(pDevIns, KBDState *));
1347 return VINF_SUCCESS;
1348}
1349
1350
1351/**
1352 * Loads a saved keyboard device state.
1353 *
1354 * @returns VBox status code.
1355 * @param pDevIns The device instance.
1356 * @param pSSMHandle The handle to the saved state.
1357 * @param u32Version The data unit version number.
1358 */
1359static DECLCALLBACK(int) kbdLoadExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSMHandle, uint32_t u32Version)
1360{
1361 return kbd_load(pSSMHandle, PDMINS2DATA(pDevIns, KBDState *), u32Version);
1362}
1363
1364/**
1365 * Reset notification.
1366 *
1367 * @returns VBox status.
1368 * @param pDevIns The device instance data.
1369 */
1370static DECLCALLBACK(void) kbdReset(PPDMDEVINS pDevIns)
1371{
1372 kbd_reset(PDMINS2DATA(pDevIns, KBDState *));
1373}
1374
1375
1376/* -=-=-=-=-=- Keyboard: IBase -=-=-=-=-=- */
1377
1378/**
1379 * Queries an interface to the driver.
1380 *
1381 * @returns Pointer to interface.
1382 * @returns NULL if the interface was not supported by the device.
1383 * @param pInterface Pointer to the keyboard port base interface (KBDState::Keyboard.Base).
1384 * @param enmInterface The requested interface identification.
1385 */
1386static DECLCALLBACK(void *) kbdKeyboardQueryInterface(PPDMIBASE pInterface, PDMINTERFACE enmInterface)
1387{
1388 KBDState *pData = (KBDState *)((uintptr_t)pInterface - RT_OFFSETOF(KBDState, Keyboard.Base));
1389 switch (enmInterface)
1390 {
1391 case PDMINTERFACE_BASE:
1392 return &pData->Keyboard.Base;
1393 case PDMINTERFACE_KEYBOARD_PORT:
1394 return &pData->Keyboard.Port;
1395 default:
1396 return NULL;
1397 }
1398}
1399
1400
1401/* -=-=-=-=-=- Keyboard: IKeyboardPort -=-=-=-=-=- */
1402
1403/** Converts a keyboard port interface pointer to a keyboard state pointer. */
1404#define IKEYBOARDPORT_2_KBDSTATE(pInterface) ( (KBDState *)((uintptr_t)pInterface - RT_OFFSETOF(KBDState, Keyboard.Port)) )
1405
1406/**
1407 * Keyboard event handler.
1408 *
1409 * @returns VBox status code.
1410 * @param pInterface Pointer to the keyboard port interface (KBDState::Keyboard.Port).
1411 * @param u8KeyCode The keycode.
1412 */
1413static DECLCALLBACK(int) kbdKeyboardPutEvent(PPDMIKEYBOARDPORT pInterface, uint8_t u8KeyCode)
1414{
1415 KBDState *pData = IKEYBOARDPORT_2_KBDSTATE(pInterface);
1416 pc_kbd_put_keycode(pData, u8KeyCode);
1417 return VINF_SUCCESS;
1418}
1419
1420
1421/* -=-=-=-=-=- Mouse: IBase -=-=-=-=-=- */
1422
1423/**
1424 * Queries an interface to the driver.
1425 *
1426 * @returns Pointer to interface.
1427 * @returns NULL if the interface was not supported by the device.
1428 * @param pInterface Pointer to the mouse port base interface (KBDState::Mouse.Base).
1429 * @param enmInterface The requested interface identification.
1430 */
1431static DECLCALLBACK(void *) kbdMouseQueryInterface(PPDMIBASE pInterface, PDMINTERFACE enmInterface)
1432{
1433 KBDState *pData = (KBDState *)((uintptr_t)pInterface - RT_OFFSETOF(KBDState, Mouse.Base));
1434 switch (enmInterface)
1435 {
1436 case PDMINTERFACE_BASE:
1437 return &pData->Mouse.Base;
1438 case PDMINTERFACE_MOUSE_PORT:
1439 return &pData->Mouse.Port;
1440 default:
1441 return NULL;
1442 }
1443}
1444
1445
1446/* -=-=-=-=-=- Mouse: IMousePort -=-=-=-=-=- */
1447
1448/** Converts a mouse port interface pointer to a keyboard state pointer. */
1449#define IMOUSEPORT_2_KBDSTATE(pInterface) ( (KBDState *)((uintptr_t)pInterface - RT_OFFSETOF(KBDState, Mouse.Port)) )
1450
1451/**
1452 * Mouse event handler.
1453 *
1454 * @returns VBox status code.
1455 * @param pInterface Pointer to the mouse port interface (KBDState::Mouse.Port).
1456 * @param i32DeltaX The X delta.
1457 * @param i32DeltaY The Y delta.
1458 * @param i32DeltaZ The Z delta.
1459 * @param fButtonStates The button states.
1460 */
1461static DECLCALLBACK(int) kbdMousePutEvent(PPDMIMOUSEPORT pInterface, int32_t i32DeltaX, int32_t i32DeltaY, int32_t i32DeltaZ, uint32_t fButtonStates)
1462{
1463 KBDState *pData = IMOUSEPORT_2_KBDSTATE(pInterface);
1464 pc_kbd_mouse_event(pData, i32DeltaX, i32DeltaY, i32DeltaZ, fButtonStates);
1465 return VINF_SUCCESS;
1466}
1467
1468
1469/* -=-=-=-=-=- real code -=-=-=-=-=- */
1470
1471
1472/**
1473 * Attach command.
1474 *
1475 * This is called to let the device attach to a driver for a specified LUN
1476 * during runtime. This is not called during VM construction, the device
1477 * constructor have to attach to all the available drivers.
1478 *
1479 * This is like plugging in the keyboard or mouse after turning on the PC.
1480 *
1481 * @returns VBox status code.
1482 * @param pDevIns The device instance.
1483 * @param iLUN The logical unit which is being detached.
1484 * @remark The keyboard controller doesn't support this action, this is just
1485 * implemented to try out the driver<->device structure.
1486 */
1487static DECLCALLBACK(int) kbdAttach(PPDMDEVINS pDevIns, unsigned iLUN)
1488{
1489 int rc;
1490 KBDState *pData = PDMINS2DATA(pDevIns, KBDState *);
1491 switch (iLUN)
1492 {
1493 /* LUN #0: keyboard */
1494 case 0:
1495 rc = pDevIns->pDevHlp->pfnDriverAttach(pDevIns, iLUN, &pData->Keyboard.Base, &pData->Keyboard.pDrvBase, "Keyboard Port");
1496 if (VBOX_SUCCESS(rc))
1497 {
1498 pData->Keyboard.pDrv = pData->Keyboard.pDrvBase->pfnQueryInterface(pData->Keyboard.pDrvBase, PDMINTERFACE_KEYBOARD_CONNECTOR);
1499 if (!pData->Keyboard.pDrv)
1500 {
1501 AssertMsgFailed(("LUN #0 doesn't have a keyboard interface! rc=%Vrc\n", rc));
1502 rc = VERR_PDM_MISSING_INTERFACE;
1503 }
1504 }
1505 else if (rc == VERR_PDM_NO_ATTACHED_DRIVER)
1506 {
1507 Log(("%s/%d: warning: no driver attached to LUN #0!\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
1508 rc = VINF_SUCCESS;
1509 }
1510 else
1511 AssertMsgFailed(("Failed to attach LUN #0! rc=%Vrc\n", rc));
1512 break;
1513
1514 /* LUN #1: aux/mouse */
1515 case 1:
1516 rc = pDevIns->pDevHlp->pfnDriverAttach(pDevIns, iLUN, &pData->Mouse.Base, &pData->Mouse.pDrvBase, "Aux (Mouse) Port");
1517 if (VBOX_SUCCESS(rc))
1518 {
1519 pData->Mouse.pDrv = pData->Mouse.pDrvBase->pfnQueryInterface(pData->Mouse.pDrvBase, PDMINTERFACE_MOUSE_CONNECTOR);
1520 if (!pData->Mouse.pDrv)
1521 {
1522 AssertMsgFailed(("LUN #1 doesn't have a mouse interface! rc=%Vrc\n", rc));
1523 rc = VERR_PDM_MISSING_INTERFACE;
1524 }
1525 }
1526 else if (rc == VERR_PDM_NO_ATTACHED_DRIVER)
1527 {
1528 Log(("%s/%d: warning: no driver attached to LUN #1!\n", pDevIns->pDevReg->szDeviceName, pDevIns->iInstance));
1529 rc = VINF_SUCCESS;
1530 }
1531 else
1532 AssertMsgFailed(("Failed to attach LUN #1! rc=%Vrc\n", rc));
1533 break;
1534
1535 default:
1536 AssertMsgFailed(("Invalid LUN #%d\n", iLUN));
1537 return VERR_PDM_NO_SUCH_LUN;
1538 }
1539
1540 return rc;
1541}
1542
1543
1544/**
1545 * Detach notification.
1546 *
1547 * This is called when a driver is detaching itself from a LUN of the device.
1548 * The device should adjust it's state to reflect this.
1549 *
1550 * This is like unplugging the network cable to use it for the laptop or
1551 * something while the PC is still running.
1552 *
1553 * @param pDevIns The device instance.
1554 * @param iLUN The logical unit which is being detached.
1555 * @remark The keyboard controller doesn't support this action, this is just
1556 * implemented to try out the driver<->device structure.
1557 */
1558static DECLCALLBACK(void) kbdDetach(PPDMDEVINS pDevIns, unsigned iLUN)
1559{
1560#if 0
1561 /*
1562 * Reset the interfaces and update the controller state.
1563 */
1564 KBDState *pData = PDMINS2DATA(pDevIns, KBDState *);
1565 switch (iLUN)
1566 {
1567 /* LUN #0: keyboard */
1568 case 0:
1569 pData->Keyboard.pDrv = NULL;
1570 pData->Keyboard.pDrvBase = NULL;
1571 break;
1572
1573 /* LUN #1: aux/mouse */
1574 case 1:
1575 pData->Mouse.pDrv = NULL;
1576 pData->Mouse.pDrvBase = NULL;
1577 break;
1578
1579 default:
1580 AssertMsgFailed(("Invalid LUN #%d\n", iLUN));
1581 break;
1582 }
1583#endif
1584}
1585
1586/**
1587 * @copydoc FNPDMDEVRELOCATE
1588 */
1589static DECLCALLBACK(void) kdbRelocate(PPDMDEVINS pDevIns, RTGCINTPTR offDelta)
1590{
1591 KBDState *pData = PDMINS2DATA(pDevIns, KBDState *);
1592 pData->pDevInsGC = PDMDEVINS_2_GCPTR(pDevIns);
1593}
1594
1595
1596/**
1597 * Construct a device instance for a VM.
1598 *
1599 * @returns VBox status.
1600 * @param pDevIns The device instance data.
1601 * If the registration structure is needed, pDevIns->pDevReg points to it.
1602 * @param iInstance Instance number. Use this to figure out which registers and such to use.
1603 * The device number is also found in pDevIns->iInstance, but since it's
1604 * likely to be freqently used PDM passes it as parameter.
1605 * @param pCfgHandle Configuration node handle for the device. Use this to obtain the configuration
1606 * of the device instance. It's also found in pDevIns->pCfgHandle, but like
1607 * iInstance it's expected to be used a bit in this function.
1608 */
1609static DECLCALLBACK(int) kbdConstruct(PPDMDEVINS pDevIns, int iInstance, PCFGMNODE pCfgHandle)
1610{
1611 KBDState *pData = PDMINS2DATA(pDevIns, KBDState *);
1612 int rc;
1613 bool fGCEnabled;
1614 bool fR0Enabled;
1615 Assert(iInstance == 0);
1616
1617 /*
1618 * Validate and read the configuration.
1619 */
1620 if (!CFGMR3AreValuesValid(pCfgHandle, "GCEnabled\0R0Enabled\0"))
1621 return VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES;
1622 rc = CFGMR3QueryBool(pCfgHandle, "GCEnabled", &fGCEnabled);
1623 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1624 fGCEnabled = true;
1625 else if (VBOX_FAILURE(rc))
1626 {
1627 AssertMsgFailed(("configuration error: failed to read GCEnabled as boolean. rc=%Vrc\n", rc));
1628 return rc;
1629 }
1630 rc = CFGMR3QueryBool(pCfgHandle, "R0Enabled", &fR0Enabled);
1631 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1632 fR0Enabled = true;
1633 else if (VBOX_FAILURE(rc))
1634 {
1635 AssertMsgFailed(("configuration error: failed to read R0Enabled as boolean. rc=%Vrc\n", rc));
1636 return rc;
1637 }
1638 Log(("pckbd: fGCEnabled=%d fR0Enabled=%d\n", fGCEnabled, fR0Enabled));
1639
1640
1641 /*
1642 * Initialize the interfaces.
1643 */
1644 pData->pDevInsHC = pDevIns;
1645 pData->pDevInsGC = PDMDEVINS_2_GCPTR(pDevIns);
1646 pData->Keyboard.Base.pfnQueryInterface = kbdKeyboardQueryInterface;
1647 pData->Keyboard.Port.pfnPutEvent = kbdKeyboardPutEvent;
1648
1649 pData->Mouse.Base.pfnQueryInterface = kbdMouseQueryInterface;
1650 pData->Mouse.Port.pfnPutEvent = kbdMousePutEvent;
1651
1652 /*
1653 * Register I/O ports, save state, keyboard event handler and mouse event handlers.
1654 */
1655 rc = PDMDevHlpIOPortRegister(pDevIns, 0x60, 1, NULL, kbdIOPortDataWrite, kbdIOPortDataRead, NULL, NULL, "PC Keyboard - Data");
1656 if (VBOX_FAILURE(rc))
1657 return rc;
1658 rc = PDMDevHlpIOPortRegister(pDevIns, 0x64, 1, NULL, kbdIOPortCommandWrite, kbdIOPortStatusRead, NULL, NULL, "PC Keyboard - Command / Status");
1659 if (VBOX_FAILURE(rc))
1660 return rc;
1661 if (fGCEnabled)
1662 {
1663 rc = PDMDevHlpIOPortRegisterGC(pDevIns, 0x60, 1, 0, "kbdIOPortDataWrite", "kbdIOPortDataRead", NULL, NULL, "PC Keyboard - Data");
1664 if (VBOX_FAILURE(rc))
1665 return rc;
1666 rc = PDMDevHlpIOPortRegisterGC(pDevIns, 0x64, 1, 0, "kbdIOPortCommandWrite", "kbdIOPortStatusRead", NULL, NULL, "PC Keyboard - Command / Status");
1667 if (VBOX_FAILURE(rc))
1668 return rc;
1669 }
1670 if (fR0Enabled)
1671 {
1672 rc = pDevIns->pDevHlp->pfnIOPortRegisterR0(pDevIns, 0x60, 1, 0, "kbdIOPortDataWrite", "kbdIOPortDataRead", NULL, NULL, "PC Keyboard - Data");
1673 if (VBOX_FAILURE(rc))
1674 return rc;
1675 rc = pDevIns->pDevHlp->pfnIOPortRegisterR0(pDevIns, 0x64, 1, 0, "kbdIOPortCommandWrite", "kbdIOPortStatusRead", NULL, NULL, "PC Keyboard - Command / Status");
1676 if (VBOX_FAILURE(rc))
1677 return rc;
1678 }
1679 rc = PDMDevHlpSSMRegister(pDevIns, g_DevicePS2KeyboardMouse.szDeviceName, iInstance, PCKBD_SAVED_STATE_VERSION, sizeof(*pData),
1680 NULL, kbdSaveExec, NULL,
1681 NULL, kbdLoadExec, NULL);
1682 if (VBOX_FAILURE(rc))
1683 return rc;
1684
1685 /*
1686 * Attach to the keyboard and mouse drivers.
1687 */
1688 rc = kbdAttach(pDevIns, 0 /* keyboard LUN # */);
1689 if (VBOX_FAILURE(rc))
1690 return rc;
1691 rc = kbdAttach(pDevIns, 1 /* aux/mouse LUN # */);
1692 if (VBOX_FAILURE(rc))
1693 return rc;
1694
1695 /*
1696 * Initialize the device state.
1697 */
1698 kbdReset(pDevIns);
1699
1700 return VINF_SUCCESS;
1701}
1702
1703
1704/**
1705 * The device registration structure.
1706 */
1707const PDMDEVREG g_DevicePS2KeyboardMouse =
1708{
1709 /* u32Version */
1710 PDM_DEVREG_VERSION,
1711 /* szDeviceName */
1712 "pckbd",
1713 /* szGCMod */
1714 "VBoxDDGC.gc",
1715 /* szR0Mod */
1716 "VBoxDDR0.r0",
1717 /* pszDescription */
1718 "PS/2 Keyboard and Mouse device. Emulates both the keyboard, mouse and the keyboard controller. "
1719 "LUN #0 is the keyboard connector. "
1720 "LUN #1 is the aux/mouse connector.",
1721 /* fFlags */
1722 PDM_DEVREG_FLAGS_HOST_BITS_DEFAULT | PDM_DEVREG_FLAGS_GUEST_BITS_32_64 | PDM_DEVREG_FLAGS_PAE36 | PDM_DEVREG_FLAGS_GC | PDM_DEVREG_FLAGS_R0,
1723 /* fClass */
1724 PDM_DEVREG_CLASS_INPUT,
1725 /* cMaxInstances */
1726 1,
1727 /* cbInstance */
1728 sizeof(KBDState),
1729 /* pfnConstruct */
1730 kbdConstruct,
1731 /* pfnDestruct */
1732 NULL,
1733 /* pfnRelocate */
1734 kdbRelocate,
1735 /* pfnIOCtl */
1736 NULL,
1737 /* pfnPowerOn */
1738 NULL,
1739 /* pfnReset */
1740 kbdReset,
1741 /* pfnSuspend */
1742 NULL,
1743 /* pfnResume */
1744 NULL,
1745 /* pfnAttach */
1746 kbdAttach,
1747 /* pfnDetach */
1748 kbdDetach,
1749 /* pfnQueryInterface. */
1750 NULL
1751};
1752
1753#endif /* IN_RING3 */
1754#endif /* VBOX */
1755#endif /* !VBOX_DEVICE_STRUCT_TESTCASE */
1756
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette