VirtualBox

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

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

New and improved PS/2 keyboard emulation.

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