VirtualBox

source: vbox/trunk/src/VBox/Devices/Input/DevPS2.cpp@ 27491

Last change on this file since 27491 was 27463, checked in by vboxsync, 15 years ago

Devices/Input/DevPS2: cleaned up some harmless messed up code and prevented harmful mouse wheel reporting if the wheel is not enabled

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