VirtualBox

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

Last change on this file since 39518 was 39091, checked in by vboxsync, 13 years ago

More parameter warning fixes; made PciIch9 check the saved state version.

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