VirtualBox

source: vbox/trunk/src/VBox/Devices/Input/PS2K.cpp@ 81677

Last change on this file since 81677 was 81216, checked in by vboxsync, 5 years ago

Bracing for more.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 67.7 KB
Line 
1/* $Id: PS2K.cpp 81216 2019-10-10 13:44:39Z vboxsync $ */
2/** @file
3 * PS2K - PS/2 keyboard emulation.
4 */
5
6/*
7 * Copyright (C) 2007-2019 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/*
19 * References:
20 *
21 * IBM PS/2 Technical Reference, Keyboards (101- and 102-Key), 1990
22 * Keyboard Scan Code Specification, Microsoft, 2000
23 *
24 * Notes:
25 * - The keyboard never sends partial scan-code sequences; if there isn't enough
26 * room left in the buffer for the entire sequence, the keystroke is discarded
27 * and an overrun code is sent instead.
28 * - Command responses do not disturb stored keystrokes and always have priority.
29 * - Caps Lock and Scroll Lock are normal keys from the keyboard's point of view.
30 * However, Num Lock is not and the keyboard internally tracks its state.
31 * - The way Print Screen works in scan set 1/2 is totally insane.
32 * - A PS/2 keyboard can send at most 1,000 to 1,500 bytes per second. There is
33 * software which relies on that fact and assumes that a scan code can be
34 * read twice before the next scan code comes in.
35 */
36
37
38/*********************************************************************************************************************************
39* Header Files *
40*********************************************************************************************************************************/
41#define LOG_GROUP LOG_GROUP_DEV_KBD
42#include <VBox/vmm/pdmdev.h>
43#include <VBox/err.h>
44#include <iprt/assert.h>
45#include <iprt/uuid.h>
46#include "VBoxDD.h"
47#define IN_PS2K
48#include "PS2Dev.h"
49
50
51/*********************************************************************************************************************************
52* Defined Constants And Macros *
53*********************************************************************************************************************************/
54/** @name Keyboard commands sent by the system.
55 * @{ */
56#define KCMD_LEDS 0xED
57#define KCMD_ECHO 0xEE
58#define KCMD_INVALID_1 0xEF
59#define KCMD_SCANSET 0xF0
60#define KCMD_INVALID_2 0xF1
61#define KCMD_READ_ID 0xF2
62#define KCMD_RATE_DELAY 0xF3
63#define KCMD_ENABLE 0xF4
64#define KCMD_DFLT_DISABLE 0xF5
65#define KCMD_SET_DEFAULT 0xF6
66#define KCMD_ALL_TYPEMATIC 0xF7
67#define KCMD_ALL_MK_BRK 0xF8
68#define KCMD_ALL_MAKE 0xF9
69#define KCMD_ALL_TMB 0xFA
70#define KCMD_TYPE_MATIC 0xFB
71#define KCMD_TYPE_MK_BRK 0xFC
72#define KCMD_TYPE_MAKE 0xFD
73#define KCMD_RESEND 0xFE
74#define KCMD_RESET 0xFF
75/** @} */
76
77/** @name Keyboard responses sent to the system.
78 * @{ */
79#define KRSP_ID1 0xAB
80#define KRSP_ID2 0x83
81#define KRSP_BAT_OK 0xAA
82#define KRSP_BAT_FAIL 0xFC /* Also a 'release keys' signal. */
83#define KRSP_ECHO 0xEE
84#define KRSP_ACK 0xFA
85#define KRSP_RESEND 0xFE
86/** @} */
87
88/** @name HID modifier range.
89 * @{ */
90#define HID_MODIFIER_FIRST 0xE0
91#define HID_MODIFIER_LAST 0xE8
92/** @} */
93
94/** @name USB HID additional constants
95 * @{ */
96/** The highest USB usage code reported by VirtualBox. */
97#define VBOX_USB_MAX_USAGE_CODE 0xE7
98/** The size of an array needed to store all USB usage codes */
99#define VBOX_USB_USAGE_ARRAY_SIZE (VBOX_USB_MAX_USAGE_CODE + 1)
100/** USB HID Keyboard Usage Page. */
101#define USB_HID_KB_PAGE 7
102/** USB HID Consumer Control Usage Page. */
103#define USB_HID_CC_PAGE 12
104/** @} */
105
106/** @name Modifier key states. Sorted in USB HID code order.
107 * @{ */
108#define MOD_LCTRL 0x01
109#define MOD_LSHIFT 0x02
110#define MOD_LALT 0x04
111#define MOD_LGUI 0x08
112#define MOD_RCTRL 0x10
113#define MOD_RSHIFT 0x20
114#define MOD_RALT 0x40
115#define MOD_RGUI 0x80
116/** @} */
117
118/* Default typematic value. */
119#define KBD_DFL_RATE_DELAY 0x2B
120
121/* Input throttling delay in milliseconds. */
122#define KBD_THROTTLE_DELAY 1
123
124/** Define a simple PS/2 input device queue. */
125#define DEF_PS2Q_TYPE(name, size) \
126 typedef struct { \
127 uint32_t rpos; \
128 uint32_t wpos; \
129 uint32_t cUsed; \
130 uint32_t cSize; \
131 uint8_t abQueue[size]; \
132 } name
133
134/* Internal keyboard queue sizes. The input queue doesn't need to be
135 * extra huge and the command queue only needs to handle a few bytes.
136 */
137#define KBD_KEY_QUEUE_SIZE 64
138#define KBD_CMD_QUEUE_SIZE 4
139
140
141/*********************************************************************************************************************************
142* Structures and Typedefs *
143*********************************************************************************************************************************/
144
145/** Typematic state. */
146typedef enum {
147 KBD_TMS_IDLE = 0, /* No typematic key active. */
148 KBD_TMS_DELAY = 1, /* In the initial delay period. */
149 KBD_TMS_REPEAT = 2, /* Key repeating at set rate. */
150 KBD_TMS_32BIT_HACK = 0x7fffffff
151} tmatic_state_t;
152
153
154DEF_PS2Q_TYPE(KbdKeyQ, KBD_KEY_QUEUE_SIZE);
155DEF_PS2Q_TYPE(KbdCmdQ, KBD_CMD_QUEUE_SIZE);
156DEF_PS2Q_TYPE(GeneriQ, 1);
157
158/**
159 * The PS/2 keyboard instance data.
160 */
161typedef struct PS2K
162{
163 /** Pointer to parent device (keyboard controller). */
164 R3PTRTYPE(void *) pParent;
165 /** Set if keyboard is enabled ('scans' for input). */
166 bool fScanning;
167 /** Set NumLock is on. */
168 bool fNumLockOn;
169 /** Selected scan set. */
170 uint8_t u8ScanSet;
171 /** Modifier key state. */
172 uint8_t u8Modifiers;
173 /** Currently processed command (if any). */
174 uint8_t u8CurrCmd;
175 /** Status indicator (LED) state. */
176 uint8_t u8LEDs;
177 /** Selected typematic delay/rate. */
178 uint8_t u8TypematicCfg;
179 /** Usage code of current typematic key, if any. */
180 uint32_t u32TypematicKey;
181 /** Current typematic repeat state. */
182 tmatic_state_t enmTypematicState;
183 /** Buffer holding scan codes to be sent to the host. */
184 KbdKeyQ keyQ;
185 /** Command response queue (priority). */
186 KbdCmdQ cmdQ;
187 /** Currently depressed keys. */
188 uint8_t abDepressedKeys[VBOX_USB_USAGE_ARRAY_SIZE];
189 /** Typematic delay in milliseconds. */
190 unsigned uTypematicDelay;
191 /** Typematic repeat period in milliseconds. */
192 unsigned uTypematicRepeat;
193 /** Set if the throttle delay is currently active. */
194 bool fThrottleActive;
195 /** Set if the input rate should be throttled. */
196 bool fThrottleEnabled;
197
198 uint8_t Alignment0[2];
199
200 /** Command delay timer - RC Ptr. */
201 PTMTIMERRC pKbdDelayTimerRC;
202 /** Typematic timer - RC Ptr. */
203 PTMTIMERRC pKbdTypematicTimerRC;
204 /** Input throttle timer - RC Ptr. */
205 PTMTIMERRC pThrottleTimerRC;
206
207 /** The device critical section protecting everything - R3 Ptr */
208 R3PTRTYPE(PPDMCRITSECT) pCritSectR3;
209
210 /** Command delay timer - R3 Ptr. */
211 PTMTIMERR3 pKbdDelayTimerR3;
212 /** Typematic timer - R3 Ptr. */
213 PTMTIMERR3 pKbdTypematicTimerR3;
214 /** Input throttle timer - R3 Ptr. */
215 PTMTIMERR3 pThrottleTimerR3;
216
217 /** Command delay timer - R0 Ptr. */
218 PTMTIMERR0 pKbdDelayTimerR0;
219 /** Typematic timer - R0 Ptr. */
220 PTMTIMERR0 pKbdTypematicTimerR0;
221 /** Input throttle timer - R0 Ptr. */
222 PTMTIMERR0 pThrottleTimerR0;
223
224 /**
225 * Keyboard port - LUN#0.
226 *
227 * @implements PDMIBASE
228 * @implements PDMIKEYBOARDPORT
229 */
230 struct
231 {
232 /** The base interface for the keyboard port. */
233 PDMIBASE IBase;
234 /** The keyboard port base interface. */
235 PDMIKEYBOARDPORT IPort;
236
237 /** The base interface of the attached keyboard driver. */
238 R3PTRTYPE(PPDMIBASE) pDrvBase;
239 /** The keyboard interface of the attached keyboard driver. */
240 R3PTRTYPE(PPDMIKEYBOARDCONNECTOR) pDrv;
241 } Keyboard;
242} PS2K, *PPS2K;
243
244AssertCompile(PS2K_STRUCT_FILLER >= sizeof(PS2K));
245
246#ifndef VBOX_DEVICE_STRUCT_TESTCASE
247
248/* Key type flags. */
249#define KF_E0 0x01 /* E0 prefix. */
250#define KF_NB 0x02 /* No break code. */
251#define KF_GK 0x04 /* Gray navigation key. */
252#define KF_PS 0x08 /* Print Screen key. */
253#define KF_PB 0x10 /* Pause/Break key. */
254#define KF_NL 0x20 /* Num Lock key. */
255#define KF_NS 0x40 /* NumPad '/' key. */
256
257/* Scan Set 3 typematic defaults. */
258#define T_U 0x00 /* Unknown value. */
259#define T_T 0x01 /* Key is typematic. */
260#define T_M 0x02 /* Key is make only. */
261#define T_B 0x04 /* Key is make/break. */
262
263/* Special key values. */
264#define NONE 0x93 /* No PS/2 scan code returned. */
265#define UNAS 0x94 /* No PS/2 scan assigned to key. */
266#define RSVD 0x95 /* Reserved, do not use. */
267#define UNKN 0x96 /* Translation unknown. */
268
269/* Key definition structure. */
270typedef struct {
271 uint8_t makeS1; /* Set 1 make code. */
272 uint8_t makeS2; /* Set 2 make code. */
273 uint8_t makeS3; /* Set 3 make code. */
274 uint8_t keyFlags; /* Key flags. */
275 uint8_t keyMatic; /* Set 3 typematic default. */
276} key_def;
277
278
279/*********************************************************************************************************************************
280* Global Variables *
281*********************************************************************************************************************************/
282#ifdef IN_RING3
283/* USB to PS/2 conversion table for regular keys (HID Usage Page 7). */
284static const key_def aPS2Keys[] = {
285 /* 00 */ {NONE, NONE, NONE, KF_NB, T_U }, /* Key N/A: No Event */
286 /* 01 */ {0xFF, 0x00, 0x00, KF_NB, T_U }, /* Key N/A: Overrun Error */
287 /* 02 */ {0xFC, 0xFC, 0xFC, KF_NB, T_U }, /* Key N/A: POST Fail */
288 /* 03 */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key N/A: ErrorUndefined */
289 /* 04 */ {0x1E, 0x1C, 0x1C, 0, T_T }, /* Key 31: a A */
290 /* 05 */ {0x30, 0x32, 0x32, 0, T_T }, /* Key 50: b B */
291 /* 06 */ {0x2E, 0x21, 0x21, 0, T_T }, /* Key 48: c C */
292 /* 07 */ {0x20, 0x23, 0x23, 0, T_T }, /* Key 33: d D */
293 /* 08 */ {0x12, 0x24, 0x24, 0, T_T }, /* Key 19: e E */
294 /* 09 */ {0x21, 0x2B, 0x2B, 0, T_T }, /* Key 34: f F */
295 /* 0A */ {0x22, 0x34, 0x34, 0, T_T }, /* Key 35: g G */
296 /* 0B */ {0x23, 0x33, 0x33, 0, T_T }, /* Key 36: h H */
297 /* 0C */ {0x17, 0x43, 0x43, 0, T_T }, /* Key 24: i I */
298 /* 0D */ {0x24, 0x3B, 0x3B, 0, T_T }, /* Key 37: j J */
299 /* 0E */ {0x25, 0x42, 0x42, 0, T_T }, /* Key 38: k K */
300 /* 0F */ {0x26, 0x4B, 0x4B, 0, T_T }, /* Key 39: l L */
301 /* 10 */ {0x32, 0x3A, 0x3A, 0, T_T }, /* Key 52: m M */
302 /* 11 */ {0x31, 0x31, 0x31, 0, T_T }, /* Key 51: n N */
303 /* 12 */ {0x18, 0x44, 0x44, 0, T_T }, /* Key 25: o O */
304 /* 13 */ {0x19, 0x4D, 0x4D, 0, T_T }, /* Key 26: p P */
305 /* 14 */ {0x10, 0x15, 0x15, 0, T_T }, /* Key 17: q Q */
306 /* 15 */ {0x13, 0x2D, 0x2D, 0, T_T }, /* Key 20: r R */
307 /* 16 */ {0x1F, 0x1B, 0x1B, 0, T_T }, /* Key 32: s S */
308 /* 17 */ {0x14, 0x2C, 0x2C, 0, T_T }, /* Key 21: t T */
309 /* 18 */ {0x16, 0x3C, 0x3C, 0, T_T }, /* Key 23: u U */
310 /* 19 */ {0x2F, 0x2A, 0x2A, 0, T_T }, /* Key 49: v V */
311 /* 1A */ {0x11, 0x1D, 0x1D, 0, T_T }, /* Key 18: w W */
312 /* 1B */ {0x2D, 0x22, 0x22, 0, T_T }, /* Key 47: x X */
313 /* 1C */ {0x15, 0x35, 0x35, 0, T_T }, /* Key 22: y Y */
314 /* 1D */ {0x2C, 0x1A, 0x1A, 0, T_T }, /* Key 46: z Z */
315 /* 1E */ {0x02, 0x16, 0x16, 0, T_T }, /* Key 2: 1 ! */
316 /* 1F */ {0x03, 0x1E, 0x1E, 0, T_T }, /* Key 3: 2 @ */
317 /* 20 */ {0x04, 0x26, 0x26, 0, T_T }, /* Key 4: 3 # */
318 /* 21 */ {0x05, 0x25, 0x25, 0, T_T }, /* Key 5: 4 $ */
319 /* 22 */ {0x06, 0x2E, 0x2E, 0, T_T }, /* Key 6: 5 % */
320 /* 23 */ {0x07, 0x36, 0x36, 0, T_T }, /* Key 7: 6 ^ */
321 /* 24 */ {0x08, 0x3D, 0x3D, 0, T_T }, /* Key 8: 7 & */
322 /* 25 */ {0x09, 0x3E, 0x3E, 0, T_T }, /* Key 9: 8 * */
323 /* 26 */ {0x0A, 0x46, 0x46, 0, T_T }, /* Key 10: 9 ( */
324 /* 27 */ {0x0B, 0x45, 0x45, 0, T_T }, /* Key 11: 0 ) */
325 /* 28 */ {0x1C, 0x5A, 0x5A, 0, T_T }, /* Key 43: Return */
326 /* 29 */ {0x01, 0x76, 0x08, 0, T_M }, /* Key 110: Escape */
327 /* 2A */ {0x0E, 0x66, 0x66, 0, T_T }, /* Key 15: Backspace */
328 /* 2B */ {0x0F, 0x0D, 0x0D, 0, T_T }, /* Key 16: Tab */
329 /* 2C */ {0x39, 0x29, 0x29, 0, T_T }, /* Key 61: Space */
330 /* 2D */ {0x0C, 0x4E, 0x4E, 0, T_T }, /* Key 12: - _ */
331 /* 2E */ {0x0D, 0x55, 0x55, 0, T_T }, /* Key 13: = + */
332 /* 2F */ {0x1A, 0x54, 0x54, 0, T_T }, /* Key 27: [ { */
333 /* 30 */ {0x1B, 0x5B, 0x5B, 0, T_T }, /* Key 28: ] } */
334 /* 31 */ {0x2B, 0x5D, 0x5C, 0, T_T }, /* Key 29: \ | */
335 /* 32 */ {0x2B, 0x5D, 0x5D, 0, T_T }, /* Key 42: Europe 1 (Note 2) */
336 /* 33 */ {0x27, 0x4C, 0x4C, 0, T_T }, /* Key 40: ; : */
337 /* 34 */ {0x28, 0x52, 0x52, 0, T_T }, /* Key 41: ' " */
338 /* 35 */ {0x29, 0x0E, 0x0E, 0, T_T }, /* Key 1: ` ~ */
339 /* 36 */ {0x33, 0x41, 0x41, 0, T_T }, /* Key 53: , < */
340 /* 37 */ {0x34, 0x49, 0x49, 0, T_T }, /* Key 54: . > */
341 /* 38 */ {0x35, 0x4A, 0x4A, 0, T_T }, /* Key 55: / ? */
342 /* 39 */ {0x3A, 0x58, 0x14, 0, T_B }, /* Key 30: Caps Lock */
343 /* 3A */ {0x3B, 0x05, 0x07, 0, T_M }, /* Key 112: F1 */
344 /* 3B */ {0x3C, 0x06, 0x0F, 0, T_M }, /* Key 113: F2 */
345 /* 3C */ {0x3D, 0x04, 0x17, 0, T_M }, /* Key 114: F3 */
346 /* 3D */ {0x3E, 0x0C, 0x1F, 0, T_M }, /* Key 115: F4 */
347 /* 3E */ {0x3F, 0x03, 0x27, 0, T_M }, /* Key 116: F5 */
348 /* 3F */ {0x40, 0x0B, 0x2F, 0, T_M }, /* Key 117: F6 */
349 /* 40 */ {0x41, 0x83, 0x37, 0, T_M }, /* Key 118: F7 */
350 /* 41 */ {0x42, 0x0A, 0x3F, 0, T_M }, /* Key 119: F8 */
351 /* 42 */ {0x43, 0x01, 0x47, 0, T_M }, /* Key 120: F9 */
352 /* 43 */ {0x44, 0x09, 0x4F, 0, T_M }, /* Key 121: F10 */
353 /* 44 */ {0x57, 0x78, 0x56, 0, T_M }, /* Key 122: F11 */
354 /* 45 */ {0x58, 0x07, 0x5E, 0, T_M }, /* Key 123: F12 */
355 /* 46 */ {0x37, 0x7C, 0x57, KF_PS, T_M }, /* Key 124: Print Screen (Note 1) */
356 /* 47 */ {0x46, 0x7E, 0x5F, 0, T_M }, /* Key 125: Scroll Lock */
357 /* 48 */ {RSVD, RSVD, RSVD, KF_PB, T_M }, /* Key 126: Break (Ctrl-Pause) */
358 /* 49 */ {0x52, 0x70, 0x67, KF_GK, T_M }, /* Key 75: Insert (Note 1) */
359 /* 4A */ {0x47, 0x6C, 0x6E, KF_GK, T_M }, /* Key 80: Home (Note 1) */
360 /* 4B */ {0x49, 0x7D, 0x6F, KF_GK, T_M }, /* Key 85: Page Up (Note 1) */
361 /* 4C */ {0x53, 0x71, 0x64, KF_GK, T_T }, /* Key 76: Delete (Note 1) */
362 /* 4D */ {0x4F, 0x69, 0x65, KF_GK, T_M }, /* Key 81: End (Note 1) */
363 /* 4E */ {0x51, 0x7A, 0x6D, KF_GK, T_M }, /* Key 86: Page Down (Note 1) */
364 /* 4F */ {0x4D, 0x74, 0x6A, KF_GK, T_T }, /* Key 89: Right Arrow (Note 1) */
365 /* 50 */ {0x4B, 0x6B, 0x61, KF_GK, T_T }, /* Key 79: Left Arrow (Note 1) */
366 /* 51 */ {0x50, 0x72, 0x60, KF_GK, T_T }, /* Key 84: Down Arrow (Note 1) */
367 /* 52 */ {0x48, 0x75, 0x63, KF_GK, T_T }, /* Key 83: Up Arrow (Note 1) */
368 /* 53 */ {0x45, 0x77, 0x76, KF_NL, T_M }, /* Key 90: Num Lock */
369 /* 54 */ {0x35, 0x4A, 0x77, KF_NS, T_M }, /* Key 95: Keypad / (Note 1) */
370 /* 55 */ {0x37, 0x7C, 0x7E, 0, T_M }, /* Key 100: Keypad * */
371 /* 56 */ {0x4A, 0x7B, 0x84, 0, T_M }, /* Key 105: Keypad - */
372 /* 57 */ {0x4E, 0x79, 0x7C, 0, T_T }, /* Key 106: Keypad + */
373 /* 58 */ {0x1C, 0x5A, 0x79, KF_E0, T_M }, /* Key 108: Keypad Enter */
374 /* 59 */ {0x4F, 0x69, 0x69, 0, T_M }, /* Key 93: Keypad 1 End */
375 /* 5A */ {0x50, 0x72, 0x72, 0, T_M }, /* Key 98: Keypad 2 Down */
376 /* 5B */ {0x51, 0x7A, 0x7A, 0, T_M }, /* Key 103: Keypad 3 PageDn */
377 /* 5C */ {0x4B, 0x6B, 0x6B, 0, T_M }, /* Key 92: Keypad 4 Left */
378 /* 5D */ {0x4C, 0x73, 0x73, 0, T_M }, /* Key 97: Keypad 5 */
379 /* 5E */ {0x4D, 0x74, 0x74, 0, T_M }, /* Key 102: Keypad 6 Right */
380 /* 5F */ {0x47, 0x6C, 0x6C, 0, T_M }, /* Key 91: Keypad 7 Home */
381 /* 60 */ {0x48, 0x75, 0x75, 0, T_M }, /* Key 96: Keypad 8 Up */
382 /* 61 */ {0x49, 0x7D, 0x7D, 0, T_M }, /* Key 101: Keypad 9 PageUp */
383 /* 62 */ {0x52, 0x70, 0x70, 0, T_M }, /* Key 99: Keypad 0 Insert */
384 /* 63 */ {0x53, 0x71, 0x71, 0, T_M }, /* Key 104: Keypad . Delete */
385 /* 64 */ {0x56, 0x61, 0x13, 0, T_T }, /* Key 45: Europe 2 (Note 2) */
386 /* 65 */ {0x5D, 0x2F, UNKN, KF_E0, T_U }, /* Key 129: App */
387 /* 66 */ {0x5E, 0x37, UNKN, KF_E0, T_U }, /* Key Unk: Keyboard Power */
388 /* 67 */ {0x59, 0x0F, UNKN, 0, T_U }, /* Key Unk: Keypad = */
389 /* 68 */ {0x64, 0x08, UNKN, 0, T_U }, /* Key Unk: F13 */
390 /* 69 */ {0x65, 0x10, UNKN, 0, T_U }, /* Key Unk: F14 */
391 /* 6A */ {0x66, 0x18, UNKN, 0, T_U }, /* Key Unk: F15 */
392 /* 6B */ {0x67, 0x20, UNKN, 0, T_U }, /* Key Unk: F16 */
393 /* 6C */ {0x68, 0x28, UNKN, 0, T_U }, /* Key Unk: F17 */
394 /* 6D */ {0x69, 0x30, UNKN, 0, T_U }, /* Key Unk: F18 */
395 /* 6E */ {0x6A, 0x38, UNKN, 0, T_U }, /* Key Unk: F19 */
396 /* 6F */ {0x6B, 0x40, UNKN, 0, T_U }, /* Key Unk: F20 */
397 /* 70 */ {0x6C, 0x48, UNKN, 0, T_U }, /* Key Unk: F21 */
398 /* 71 */ {0x6D, 0x50, UNKN, 0, T_U }, /* Key Unk: F22 */
399 /* 72 */ {0x6E, 0x57, UNKN, 0, T_U }, /* Key Unk: F23 */
400 /* 73 */ {0x76, 0x5F, UNKN, 0, T_U }, /* Key Unk: F24 */
401 /* 74 */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Execute */
402 /* 75 */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Help */
403 /* 76 */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Menu */
404 /* 77 */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Select */
405 /* 78 */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Stop */
406 /* 79 */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Again */
407 /* 7A */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Undo */
408 /* 7B */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Cut */
409 /* 7C */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Copy */
410 /* 7D */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Paste */
411 /* 7E */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Find */
412 /* 7F */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Mute */
413 /* 80 */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Volume Up */
414 /* 81 */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Volume Dn */
415 /* 82 */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Locking Caps Lock */
416 /* 83 */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Locking Num Lock */
417 /* 84 */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Locking Scroll Lock */
418 /* 85 */ {0x7E, 0x6D, UNKN, 0, T_U }, /* Key Unk: Keypad , (Brazilian Keypad .) */
419 /* 86 */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Equal Sign */
420 /* 87 */ {0x73, 0x51, UNKN, 0, T_U }, /* Key Unk: Keyboard Intl 1 (Ro) */
421 /* 88 */ {0x70, 0x13, UNKN, 0, T_U }, /* Key Unk: Keyboard Intl2 (K'kana/H'gana) */
422 /* 89 */ {0x7D, 0x6A, UNKN, 0, T_U }, /* Key Unk: Keyboard Intl 2 (Yen) */
423 /* 8A */ {0x79, 0x64, UNKN, 0, T_U }, /* Key Unk: Keyboard Intl 4 (Henkan) */
424 /* 8B */ {0x7B, 0x67, UNKN, 0, T_U }, /* Key Unk: Keyboard Intl 5 (Muhenkan) */
425 /* 8C */ {0x5C, 0x27, UNKN, 0, T_U }, /* Key Unk: Keyboard Intl 6 (PC9800 Pad ,) */
426 /* 8D */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Intl 7 */
427 /* 8E */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Intl 8 */
428 /* 8F */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Intl 9 */
429 /* 90 */ {0xF2, 0xF2, UNKN, KF_NB, T_U }, /* Key Unk: Keyboard Lang 1 (Hang'l/Engl) */
430 /* 91 */ {0xF1, 0xF1, UNKN, KF_NB, T_U }, /* Key Unk: Keyboard Lang 2 (Hanja) */
431 /* 92 */ {0x78, 0x63, UNKN, 0, T_U }, /* Key Unk: Keyboard Lang 3 (Katakana) */
432 /* 93 */ {0x77, 0x62, UNKN, 0, T_U }, /* Key Unk: Keyboard Lang 4 (Hiragana) */
433 /* 94 */ {0x76, 0x5F, UNKN, 0, T_U }, /* Key Unk: Keyboard Lang 5 (Zen/Han) */
434 /* 95 */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Lang 6 */
435 /* 96 */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Lang 7 */
436 /* 97 */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Lang 8 */
437 /* 98 */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Lang 9 */
438 /* 99 */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Alternate Erase */
439 /* 9A */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard SysReq/Attention (Note 3) */
440 /* 9B */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Cancel */
441 /* 9C */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Clear */
442 /* 9D */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Prior */
443 /* 9E */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Return */
444 /* 9F */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Separator */
445 /* A0 */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Out */
446 /* A1 */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Oper */
447 /* A2 */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard Clear/Again */
448 /* A3 */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard CrSel/Props */
449 /* A4 */ {UNAS, UNAS, UNAS, 0, T_U }, /* Key Unk: Keyboard ExSel */
450};
451
452/*
453 * Note 1: The behavior of these keys depends on the state of modifier keys
454 * at the time the key was pressed.
455 *
456 * Note 2: The key label depends on the national version of the keyboard.
457 *
458 * Note 3: Certain keys which have their own PS/2 scancodes do not exist on
459 * USB keyboards; the SysReq key is an example. The SysReq key scancode needs
460 * to be translated to the Print Screen HID usage code. The HID usage to PS/2
461 * scancode conversion then generates the correct sequence depending on the
462 * keyboard state.
463 */
464
465/* USB to PS/2 conversion table for modifier keys (HID Usage Page 7). */
466static const key_def aPS2ModKeys[] = {
467 /* E0 */ {0x1D, 0x14, 0x11, 0, T_B }, /* Key 58: Left Control */
468 /* E1 */ {0x2A, 0x12, 0x12, 0, T_B }, /* Key 44: Left Shift */
469 /* E2 */ {0x38, 0x11, 0x19, 0, T_B }, /* Key 60: Left Alt */
470 /* E3 */ {0x5B, 0x1F, UNKN, KF_E0, T_U }, /* Key 127: Left GUI */
471 /* E4 */ {0x1D, 0x14, 0x58, KF_E0, T_M }, /* Key 64: Right Control */
472 /* E5 */ {0x36, 0x59, 0x59, 0, T_B }, /* Key 57: Right Shift */
473 /* E6 */ {0x38, 0x11, 0x39, KF_E0, T_M }, /* Key 62: Right Alt */
474 /* E7 */ {0x5C, 0x27, UNKN, KF_E0, T_U }, /* Key 128: Right GUI */
475};
476
477/* Extended key definition for sparse mapping. */
478typedef struct {
479 uint16_t usageId;
480 key_def kdef;
481} ext_key_def;
482
483
484/* USB to PS/2 conversion table for consumer control keys (HID Usage Page 12). */
485/* This usage page is very sparse so we'll just search through it. */
486static const ext_key_def aPS2CCKeys[] = {
487 {0x00B5, {0x19, 0x4D, UNKN, KF_E0, T_U}}, /* Scan Next Track */
488 {0x00B6, {0x10, 0x15, UNKN, KF_E0, T_U}}, /* Scan Previous Track */
489 {0x00B7, {0x24, 0x3B, UNKN, KF_E0, T_U}}, /* Stop */
490 {0x00CD, {0x22, 0x34, UNKN, KF_E0, T_U}}, /* Play/Pause */
491 {0x00E2, {0x20, 0x23, UNKN, KF_E0, T_U}}, /* Mute */
492 {0x00E5, {UNAS, UNAS, UNAS, 0, T_U}}, /* Bass Boost */
493 {0x00E7, {UNAS, UNAS, UNAS, 0, T_U}}, /* Loudness */
494 {0x00E9, {0x30, 0x32, UNKN, KF_E0, T_U}}, /* Volume Up */
495 {0x00EA, {0x2E, 0x21, UNKN, KF_E0, T_U}}, /* Volume Down */
496 {0x0152, {UNAS, UNAS, UNAS, 0, T_U}}, /* Bass Up */
497 {0x0153, {UNAS, UNAS, UNAS, 0, T_U}}, /* Bass Down */
498 {0x0154, {UNAS, UNAS, UNAS, 0, T_U}}, /* Treble Up */
499 {0x0155, {UNAS, UNAS, UNAS, 0, T_U}}, /* Treble Down */
500 {0x0183, {0x6D, 0x50, UNKN, KF_E0, T_U}}, /* Media Select */
501 {0x018A, {0x6C, 0x48, UNKN, KF_E0, T_U}}, /* Mail */
502 {0x0192, {0x21, 0x2B, UNKN, KF_E0, T_U}}, /* Calculator */
503 {0x0194, {0x6B, 0x40, UNKN, KF_E0, T_U}}, /* My Computer */
504 {0x0221, {0x65, 0x10, UNKN, KF_E0, T_U}}, /* WWW Search */
505 {0x0223, {0x32, 0x3A, UNKN, KF_E0, T_U}}, /* WWW Home */
506 {0x0224, {0x6A, 0x38, UNKN, KF_E0, T_U}}, /* WWW Back */
507 {0x0225, {0x69, 0x30, UNKN, KF_E0, T_U}}, /* WWW Forward */
508 {0x0226, {0x68, 0x28, UNKN, KF_E0, T_U}}, /* WWW Stop */
509 {0x0227, {0x67, 0x20, UNKN, KF_E0, T_U}}, /* WWW Refresh */
510 {0x022A, {0x66, 0x18, UNKN, KF_E0, T_U}}, /* WWW Favorites */
511};
512
513#endif /* IN_RING3 */
514
515
516
517/**
518 * Clear a queue.
519 *
520 * @param pQ Pointer to the queue.
521 */
522static void ps2kClearQueue(GeneriQ *pQ)
523{
524 LogFlowFunc(("Clearing queue %p\n", pQ));
525 pQ->wpos = pQ->rpos;
526 pQ->cUsed = 0;
527}
528
529
530/**
531 * Add a byte to a queue.
532 *
533 * @param pQ Pointer to the queue.
534 * @param val The byte to store.
535 */
536static void ps2kInsertQueue(GeneriQ *pQ, uint8_t val)
537{
538 /* Check if queue is full. */
539 if (pQ->cUsed >= pQ->cSize)
540 {
541 LogRelFlowFunc(("queue %p full (%d entries)\n", pQ, pQ->cUsed));
542 return;
543 }
544 /* Insert data and update circular buffer write position. */
545 pQ->abQueue[pQ->wpos] = val;
546 if (++pQ->wpos == pQ->cSize)
547 pQ->wpos = 0; /* Roll over. */
548 ++pQ->cUsed;
549 LogRelFlowFunc(("inserted 0x%02X into queue %p\n", val, pQ));
550}
551
552#ifdef IN_RING3
553
554/**
555 * Add a null-terminated byte sequence to a queue if there is enough room.
556 *
557 * @param pQ Pointer to the queue.
558 * @param pStr Pointer to the bytes to store.
559 * @param uReserve Number of bytes that must still remain
560 * available in queue.
561 * @return int VBox status/error code.
562 */
563static int ps2kInsertStrQueue(GeneriQ *pQ, const uint8_t *pStr, uint32_t uReserve)
564{
565 uint32_t cbStr;
566 unsigned i;
567
568 cbStr = (uint32_t)strlen((const char *)pStr);
569
570 /* Check if queue has enough room. */
571 if (pQ->cUsed + uReserve + cbStr >= pQ->cSize)
572 {
573 LogRelFlowFunc(("queue %p full (%u entries, want room for %u), cannot insert %u entries\n",
574 pQ, pQ->cUsed, uReserve, cbStr));
575 return VERR_BUFFER_OVERFLOW;
576 }
577
578 /* Insert byte sequence and update circular buffer write position. */
579 for (i = 0; i < cbStr; ++i) {
580 pQ->abQueue[pQ->wpos] = pStr[i];
581 if (++pQ->wpos == pQ->cSize)
582 pQ->wpos = 0; /* Roll over. */
583 }
584 pQ->cUsed += cbStr;
585 LogRelFlowFunc(("inserted %u bytes into queue %p\n", cbStr, pQ));
586 return VINF_SUCCESS;
587}
588
589/**
590 * Save a queue state.
591 *
592 * @param pSSM SSM handle to write the state to.
593 * @param pQ Pointer to the queue.
594 */
595static void ps2kSaveQueue(PSSMHANDLE pSSM, GeneriQ *pQ)
596{
597 uint32_t cItems = pQ->cUsed;
598 int i;
599
600 /* Only save the number of items. Note that the read/write
601 * positions aren't saved as they will be rebuilt on load.
602 */
603 SSMR3PutU32(pSSM, cItems);
604
605 LogFlow(("Storing %d items from queue %p\n", cItems, pQ));
606
607 /* Save queue data - only the bytes actually used (typically zero). */
608 for (i = pQ->rpos; cItems-- > 0; i = (i + 1) % pQ->cSize)
609 SSMR3PutU8(pSSM, pQ->abQueue[i]);
610}
611
612/**
613 * Load a queue state.
614 *
615 * @param pSSM SSM handle to read the state from.
616 * @param pQ Pointer to the queue.
617 *
618 * @return int VBox status/error code.
619 */
620static int ps2kLoadQueue(PSSMHANDLE pSSM, GeneriQ *pQ)
621{
622 /* On load, always put the read pointer at zero. */
623 int rc = SSMR3GetU32(pSSM, &pQ->cUsed);
624 AssertRCReturn(rc, rc);
625 LogFlow(("Loading %u items to queue %p\n", pQ->cUsed, pQ));
626 AssertMsgReturn(pQ->cUsed <= pQ->cSize, ("Saved size=%u, actual=%u\n", pQ->cUsed, pQ->cSize),
627 VERR_SSM_DATA_UNIT_FORMAT_CHANGED);
628
629 /* Recalculate queue positions and load data in one go. */
630 pQ->rpos = 0;
631 pQ->wpos = pQ->cUsed;
632 rc = SSMR3GetMem(pSSM, pQ->abQueue, pQ->cUsed);
633
634 return rc;
635}
636
637/**
638 * Notify listener about LEDs state change.
639 *
640 * @param pThis The PS/2 keyboard instance data.
641 * @param u8State Bitfield which reflects LEDs state.
642 */
643static void ps2kNotifyLedsState(PPS2K pThis, uint8_t u8State)
644{
645
646 PDMKEYBLEDS enmLeds = PDMKEYBLEDS_NONE;
647
648 if (u8State & 0x01)
649 enmLeds = (PDMKEYBLEDS)(enmLeds | PDMKEYBLEDS_SCROLLLOCK);
650 if (u8State & 0x02)
651 enmLeds = (PDMKEYBLEDS)(enmLeds | PDMKEYBLEDS_NUMLOCK);
652 if (u8State & 0x04)
653 enmLeds = (PDMKEYBLEDS)(enmLeds | PDMKEYBLEDS_CAPSLOCK);
654
655 pThis->Keyboard.pDrv->pfnLedStatusChange(pThis->Keyboard.pDrv, enmLeds);
656
657}
658
659/**
660 * Query the number of items currently in a queue.
661 *
662 * @param pQ Pointer to the queue.
663 *
664 * @return uint32_t Number of items in queue.
665 */
666static uint32_t ps2kInQueue(GeneriQ *pQ)
667{
668 return pQ->cUsed;
669}
670
671#endif /* IN_RING3 */
672
673/**
674 * Retrieve a byte from a queue.
675 *
676 * @param pQ Pointer to the queue.
677 * @param pVal Pointer to storage for the byte.
678 *
679 * @return int VINF_TRY_AGAIN if queue is empty,
680 * VINF_SUCCESS if a byte was read.
681 */
682static int ps2kRemoveQueue(GeneriQ *pQ, uint8_t *pVal)
683{
684 int rc = VINF_TRY_AGAIN;
685
686 Assert(pVal);
687 if (pQ->cUsed)
688 {
689 *pVal = pQ->abQueue[pQ->rpos];
690 if (++pQ->rpos == pQ->cSize)
691 pQ->rpos = 0; /* Roll over. */
692 --pQ->cUsed;
693 rc = VINF_SUCCESS;
694 LogFlowFunc(("removed 0x%02X from queue %p\n", *pVal, pQ));
695 } else
696 LogFlowFunc(("queue %p empty\n", pQ));
697 return rc;
698}
699
700/* Clears the currently active typematic key, if any. */
701static void ps2kStopTypematicRepeat(PPS2K pThis)
702{
703 if (pThis->u32TypematicKey)
704 {
705 LogFunc(("Typematic key %08X\n", pThis->u32TypematicKey));
706 pThis->enmTypematicState = KBD_TMS_IDLE;
707 pThis->u32TypematicKey = 0;
708 TMTimerStop(pThis->CTX_SUFF(pKbdTypematicTimer));
709 }
710}
711
712/* Convert encoded typematic value to milliseconds. Note that the values are rated
713 * with +/- 20% accuracy, so there's no need for high precision.
714 */
715static void ps2kSetupTypematic(PPS2K pThis, uint8_t val)
716{
717 int A, B;
718 unsigned period;
719
720 pThis->u8TypematicCfg = val;
721 /* The delay is easy: (1 + value) * 250 ms */
722 pThis->uTypematicDelay = (1 + ((val >> 5) & 3)) * 250;
723 /* The rate is more complicated: (8 + A) * 2^B * 4.17 ms */
724 A = val & 7;
725 B = (val >> 3) & 3;
726 period = (8 + A) * (1 << B) * 417 / 100;
727 pThis->uTypematicRepeat = period;
728 Log(("Typematic delay %u ms, repeat period %u ms\n",
729 pThis->uTypematicDelay, pThis->uTypematicRepeat));
730}
731
732static void ps2kSetDefaults(PPS2K pThis)
733{
734 LogFlowFunc(("Set keyboard defaults\n"));
735 ps2kClearQueue((GeneriQ *)&pThis->keyQ);
736 /* Set default Scan Set 3 typematic values. */
737 /* Set default typematic rate/delay. */
738 ps2kSetupTypematic(pThis, KBD_DFL_RATE_DELAY);
739 /* Clear last typematic key?? */
740 ps2kStopTypematicRepeat(pThis);
741}
742
743/**
744 * Receive and process a byte sent by the keyboard controller.
745 *
746 * @param pThis The PS/2 keyboard instance data.
747 * @param cmd The command (or data) byte.
748 */
749int PS2KByteToKbd(PPS2K pThis, uint8_t cmd)
750{
751 bool fHandled = true;
752
753 LogFlowFunc(("new cmd=0x%02X, active cmd=0x%02X\n", cmd, pThis->u8CurrCmd));
754
755 if (pThis->u8CurrCmd == KCMD_RESET)
756 /* In reset mode, do not respond at all. */
757 return VINF_SUCCESS;
758
759 switch (cmd)
760 {
761 case KCMD_ECHO:
762 ps2kInsertQueue((GeneriQ *)&pThis->cmdQ, KRSP_ECHO);
763 pThis->u8CurrCmd = 0;
764 break;
765 case KCMD_READ_ID:
766 ps2kInsertQueue((GeneriQ *)&pThis->cmdQ, KRSP_ACK);
767 ps2kInsertQueue((GeneriQ *)&pThis->cmdQ, KRSP_ID1);
768 ps2kInsertQueue((GeneriQ *)&pThis->cmdQ, KRSP_ID2);
769 pThis->u8CurrCmd = 0;
770 break;
771 case KCMD_ENABLE:
772 pThis->fScanning = true;
773 ps2kClearQueue((GeneriQ *)&pThis->keyQ);
774 ps2kStopTypematicRepeat(pThis);
775 ps2kInsertQueue((GeneriQ *)&pThis->cmdQ, KRSP_ACK);
776 pThis->u8CurrCmd = 0;
777 break;
778 case KCMD_DFLT_DISABLE:
779 pThis->fScanning = false;
780 ps2kSetDefaults(pThis); /* Also clears buffer/typematic state. */
781 ps2kInsertQueue((GeneriQ *)&pThis->cmdQ, KRSP_ACK);
782 pThis->u8CurrCmd = 0;
783 break;
784 case KCMD_SET_DEFAULT:
785 ps2kSetDefaults(pThis);
786 ps2kInsertQueue((GeneriQ *)&pThis->cmdQ, KRSP_ACK);
787 pThis->u8CurrCmd = 0;
788 break;
789 case KCMD_ALL_TYPEMATIC:
790 case KCMD_ALL_MK_BRK:
791 case KCMD_ALL_MAKE:
792 case KCMD_ALL_TMB:
793 /// @todo Set the key types here.
794 ps2kInsertQueue((GeneriQ *)&pThis->cmdQ, KRSP_ACK);
795 pThis->u8CurrCmd = 0;
796 break;
797 case KCMD_RESEND:
798 pThis->u8CurrCmd = 0;
799 break;
800 case KCMD_RESET:
801 pThis->u8ScanSet = 2;
802 ps2kSetDefaults(pThis);
803 /// @todo reset more?
804 ps2kInsertQueue((GeneriQ *)&pThis->cmdQ, KRSP_ACK);
805 pThis->u8CurrCmd = cmd;
806 /* Delay BAT completion; the test may take hundreds of ms. */
807 TMTimerSetMillies(pThis->CTX_SUFF(pKbdDelayTimer), 2);
808 break;
809 /* The following commands need a parameter. */
810 case KCMD_LEDS:
811 case KCMD_SCANSET:
812 case KCMD_RATE_DELAY:
813 case KCMD_TYPE_MATIC:
814 case KCMD_TYPE_MK_BRK:
815 case KCMD_TYPE_MAKE:
816 ps2kInsertQueue((GeneriQ *)&pThis->cmdQ, KRSP_ACK);
817 pThis->u8CurrCmd = cmd;
818 break;
819 default:
820 /* Sending a command instead of a parameter starts the new command. */
821 switch (pThis->u8CurrCmd)
822 {
823 case KCMD_LEDS:
824#ifndef IN_RING3
825 return VINF_IOM_R3_IOPORT_WRITE;
826#else
827 {
828 ps2kNotifyLedsState(pThis, cmd);
829 pThis->fNumLockOn = !!(cmd & 0x02); /* Sync internal Num Lock state. */
830 ps2kInsertQueue((GeneriQ *)&pThis->cmdQ, KRSP_ACK);
831 pThis->u8LEDs = cmd;
832 pThis->u8CurrCmd = 0;
833 }
834#endif
835 break;
836 case KCMD_SCANSET:
837 ps2kInsertQueue((GeneriQ *)&pThis->cmdQ, KRSP_ACK);
838 if (cmd == 0)
839 ps2kInsertQueue((GeneriQ *)&pThis->cmdQ, pThis->u8ScanSet);
840 else if (cmd < 4)
841 {
842 pThis->u8ScanSet = cmd;
843 LogRel(("PS2K: Selected scan set %d\n", cmd));
844 }
845 /* Other values are simply ignored. */
846 pThis->u8CurrCmd = 0;
847 break;
848 case KCMD_RATE_DELAY:
849 ps2kSetupTypematic(pThis, cmd);
850 ps2kInsertQueue((GeneriQ *)&pThis->cmdQ, KRSP_ACK);
851 pThis->u8CurrCmd = 0;
852 break;
853 default:
854 fHandled = false;
855 }
856 /* Fall through only to handle unrecognized commands. */
857 if (fHandled)
858 break;
859 RT_FALL_THRU();
860
861 case KCMD_INVALID_1:
862 case KCMD_INVALID_2:
863 ps2kInsertQueue((GeneriQ *)&pThis->cmdQ, KRSP_RESEND);
864 pThis->u8CurrCmd = 0;
865 break;
866 }
867 LogFlowFunc(("Active cmd now 0x%02X; updating interrupts\n", pThis->u8CurrCmd));
868// KBCUpdateInterrupts(pThis->pParent);
869 return VINF_SUCCESS;
870}
871
872/**
873 * Send a byte (keystroke or command response) to the keyboard controller.
874 *
875 * @returns VINF_SUCCESS or VINF_TRY_AGAIN.
876 * @param pThis The PS/2 keyboard instance data.
877 * @param pb Where to return the byte we've read.
878 * @remarks Caller must have entered the device critical section.
879 */
880int PS2KByteFromKbd(PPS2K pThis, uint8_t *pb)
881{
882 int rc;
883
884 AssertPtr(pb);
885
886 /* Anything in the command queue has priority over data
887 * in the keystroke queue. Additionally, keystrokes are
888 * blocked if a command is currently in progress, even if
889 * the command queue is empty.
890 */
891 rc = ps2kRemoveQueue((GeneriQ *)&pThis->cmdQ, pb);
892 if (rc != VINF_SUCCESS && !pThis->u8CurrCmd && pThis->fScanning)
893 if (!pThis->fThrottleActive)
894 {
895 rc = ps2kRemoveQueue((GeneriQ *)&pThis->keyQ, pb);
896 if (pThis->fThrottleEnabled) {
897 pThis->fThrottleActive = true;
898 TMTimerSetMillies(pThis->CTX_SUFF(pThrottleTimer), KBD_THROTTLE_DELAY);
899 }
900 }
901
902 LogFlowFunc(("keyboard sends 0x%02x (%svalid data)\n", *pb, rc == VINF_SUCCESS ? "" : "not "));
903 return rc;
904}
905
906#ifdef IN_RING3
907
908static int ps2kProcessKeyEvent(PPS2K pThis, uint32_t u32HidCode, bool fKeyDown)
909{
910 key_def const *pKeyDef;
911 uint8_t abCodes[16];
912 char *pCodes;
913 size_t cbLeft;
914 uint8_t abScan[2];
915 uint8_t u8HidPage;
916 uint8_t u8HidCode;
917 uint16_t u16HidUsage;
918
919 u8HidPage = RT_LOBYTE(RT_HIWORD(u32HidCode));
920 u16HidUsage = RT_LOWORD(u32HidCode);
921 /* For the keyboard usage page (7) we use a 8-bit code. For other pages we use the full 16-bit ID. */
922 u8HidCode = (u8HidPage == USB_HID_KB_PAGE) ? RT_LOBYTE(u32HidCode) : 0;
923
924 LogFlowFunc(("key %s: page 0x%02x ID 0x%04x (set %d)\n", fKeyDown ? "down" : "up", u8HidPage, u16HidUsage, pThis->u8ScanSet));
925
926 /* Find the key definition in somewhat sparse storage. */
927 if (u8HidPage == USB_HID_KB_PAGE)
928 /* For the standard keyboard usage page, thre are just two arrays. */
929 pKeyDef = u8HidCode >= HID_MODIFIER_FIRST ? &aPS2ModKeys[u8HidCode - HID_MODIFIER_FIRST] : &aPS2Keys[u8HidCode];
930 else if (u8HidPage == USB_HID_CC_PAGE)
931 {
932 /* For the consumer control usage page, we need to search. */
933 unsigned i;
934
935 pKeyDef = &aPS2Keys[0]; /* Dummy no-event key. */
936 for (i = 0; i < RT_ELEMENTS(aPS2CCKeys); ++i)
937 {
938 if (aPS2CCKeys[i].usageId == u16HidUsage)
939 {
940 pKeyDef = &aPS2CCKeys[i].kdef;
941 break;
942 }
943 }
944 }
945 else
946 {
947 LogFlow(("Unsupported HID usage page, ignoring key.\n"));
948 return VINF_SUCCESS;
949 }
950
951 /* Some keys are not processed at all; early return. */
952 if (pKeyDef->makeS1 == NONE)
953 {
954 LogFlow(("Skipping key processing.\n"));
955 return VINF_SUCCESS;
956 }
957
958 /* Handle modifier keys (Ctrl/Alt/Shift/GUI). We need to keep track
959 * of their state in addition to sending the scan code.
960 */
961 if (u8HidCode >= HID_MODIFIER_FIRST)
962 {
963 unsigned mod_bit = 1 << (u8HidCode - HID_MODIFIER_FIRST);
964
965 Assert((u8HidPage == USB_HID_KB_PAGE));
966 Assert((u8HidCode <= HID_MODIFIER_LAST));
967 if (fKeyDown)
968 pThis->u8Modifiers |= mod_bit;
969 else
970 pThis->u8Modifiers &= ~mod_bit;
971 }
972
973 /* Toggle NumLock state. */
974 if ((pKeyDef->keyFlags & KF_NL) && fKeyDown)
975 pThis->fNumLockOn ^= true;
976
977 abCodes[0] = 0;
978 pCodes = (char *)abCodes;
979 cbLeft = sizeof(abCodes);
980
981 if (pThis->u8ScanSet == 1 || pThis->u8ScanSet == 2)
982 {
983 /* The basic scan set 1 and 2 logic is the same, only the scan codes differ.
984 * Since scan set 2 is used almost all the time, that case is handled first.
985 */
986 if (fKeyDown)
987 {
988 /* Process key down event. */
989 if (pKeyDef->keyFlags & KF_PB)
990 {
991 /* Pause/Break sends different data if either Ctrl is held. */
992 if (pThis->u8Modifiers & (MOD_LCTRL | MOD_RCTRL))
993 RTStrCatP(&pCodes, &cbLeft, pThis->u8ScanSet == 2 ?
994 "\xE0\x7E\xE0\xF0\x7E" : "\xE0\x46\xE0\xC6");
995 else
996 RTStrCatP(&pCodes, &cbLeft, pThis->u8ScanSet == 2 ?
997 "\xE1\x14\x77\xE1\xF0\x14\xF0\x77" : "\xE1\x1D\x45\xE1\x9D\xC5");
998 }
999 else if (pKeyDef->keyFlags & KF_PS)
1000 {
1001 /* Print Screen depends on all of Ctrl, Shift, *and* Alt! */
1002 if (pThis->u8Modifiers & (MOD_LALT | MOD_RALT))
1003 RTStrCatP(&pCodes, &cbLeft, pThis->u8ScanSet == 2 ?
1004 "\x84" : "\x54");
1005 else if (pThis->u8Modifiers & (MOD_LSHIFT | MOD_RSHIFT))
1006 RTStrCatP(&pCodes, &cbLeft, pThis->u8ScanSet == 2 ?
1007 "\xE0\x7C" : "\xE0\x37");
1008 else
1009 RTStrCatP(&pCodes, &cbLeft, pThis->u8ScanSet == 2 ?
1010 "\xE0\x12\xE0\x7C" : "\xE0\x2A\xE0\x37");
1011 }
1012 else if (pKeyDef->keyFlags & (KF_GK | KF_NS))
1013 {
1014 /* The numeric pad keys fake Shift presses or releases
1015 * depending on Num Lock and Shift key state. The '/'
1016 * key behaves in a similar manner but does not depend on
1017 * the Num Lock state.
1018 */
1019 if (!pThis->fNumLockOn || (pKeyDef->keyFlags & KF_NS))
1020 {
1021 if (pThis->u8Modifiers & MOD_LSHIFT)
1022 RTStrCatP(&pCodes, &cbLeft, pThis->u8ScanSet == 2 ?
1023 "\xE0\xF0\x12" : "\xE0\xAA");
1024 if (pThis->u8Modifiers & MOD_RSHIFT)
1025 RTStrCatP(&pCodes, &cbLeft, pThis->u8ScanSet == 2 ?
1026 "\xE0\xF0\x59" : "\xE0\xB6");
1027 }
1028 else
1029 {
1030 Assert(pThis->fNumLockOn); /* Not for KF_NS! */
1031 if ((pThis->u8Modifiers & (MOD_LSHIFT | MOD_RSHIFT)) == 0)
1032 RTStrCatP(&pCodes, &cbLeft, pThis->u8ScanSet == 2 ?
1033 "\xE0\x12" : "\xE0\x2A");
1034 /* Else Shift cancels NumLock, so no prefix! */
1035 }
1036 }
1037
1038 /* Standard processing for regular keys only. */
1039 abScan[0] = pThis->u8ScanSet == 2 ? pKeyDef->makeS2 : pKeyDef->makeS1;
1040 abScan[1] = '\0';
1041 if (!(pKeyDef->keyFlags & (KF_PB | KF_PS)))
1042 {
1043 if (pKeyDef->keyFlags & (KF_E0 | KF_GK | KF_NS))
1044 RTStrCatP(&pCodes, &cbLeft, "\xE0");
1045 RTStrCatP(&pCodes, &cbLeft, (const char *)abScan);
1046 }
1047
1048 /* Feed the bytes to the queue if there is room. */
1049 /// @todo Send overrun code if sequence won't fit?
1050 ps2kInsertStrQueue((GeneriQ *)&pThis->keyQ, abCodes, 0);
1051 }
1052 else if (!(pKeyDef->keyFlags & (KF_NB | KF_PB)))
1053 {
1054 /* Process key up event except for keys which produce none. */
1055
1056 /* Handle Print Screen release. */
1057 if (pKeyDef->keyFlags & KF_PS)
1058 {
1059 /* Undo faked Print Screen state as needed. */
1060 if (pThis->u8Modifiers & (MOD_LALT | MOD_RALT))
1061 RTStrCatP(&pCodes, &cbLeft, pThis->u8ScanSet == 2 ?
1062 "\xF0\x84" : "\xD4");
1063 else if (pThis->u8Modifiers & (MOD_LSHIFT | MOD_RSHIFT))
1064 RTStrCatP(&pCodes, &cbLeft, pThis->u8ScanSet == 2 ?
1065 "\xE0\xF0\x7C" : "\xE0\xB7");
1066 else
1067 RTStrCatP(&pCodes, &cbLeft, pThis->u8ScanSet == 2 ?
1068 "\xE0\xF0\x7C\xE0\xF0\x12" : "\xE0\xB7\xE0\xAA");
1069 }
1070 else
1071 {
1072 /* Process base scan code for less unusual keys. */
1073 abScan[0] = pThis->u8ScanSet == 2 ? pKeyDef->makeS2 : pKeyDef->makeS1 | 0x80;
1074 abScan[1] = '\0';
1075 if (pKeyDef->keyFlags & (KF_E0 | KF_GK | KF_NS))
1076 RTStrCatP(&pCodes, &cbLeft, "\xE0");
1077 if (pThis->u8ScanSet == 2)
1078 RTStrCatP(&pCodes, &cbLeft, "\xF0");
1079 RTStrCatP(&pCodes, &cbLeft, (const char *)abScan);
1080
1081 /* Restore shift state for gray keys. */
1082 if (pKeyDef->keyFlags & (KF_GK | KF_NS))
1083 {
1084 if (!pThis->fNumLockOn || (pKeyDef->keyFlags & KF_NS))
1085 {
1086 if (pThis->u8Modifiers & MOD_LSHIFT)
1087 RTStrCatP(&pCodes, &cbLeft, pThis->u8ScanSet == 2 ?
1088 "\xE0\x12" : "\xE0\x2A");
1089 if (pThis->u8Modifiers & MOD_RSHIFT)
1090 RTStrCatP(&pCodes, &cbLeft, pThis->u8ScanSet == 2 ?
1091 "\xE0\x59" : "\xE0\x36");
1092 }
1093 else
1094 {
1095 Assert(pThis->fNumLockOn); /* Not for KF_NS! */
1096 if ((pThis->u8Modifiers & (MOD_LSHIFT | MOD_RSHIFT)) == 0)
1097 RTStrCatP(&pCodes, &cbLeft, pThis->u8ScanSet == 2 ?
1098 "\xE0\xF0\x12" : "\xE0\xAA");
1099 }
1100 }
1101 }
1102
1103 /* Feed the bytes to the queue if there is room. */
1104 /// @todo Send overrun code if sequence won't fit?
1105 ps2kInsertStrQueue((GeneriQ *)&pThis->keyQ, abCodes, 0);
1106 }
1107 }
1108 else
1109 {
1110 /* Handle Scan Set 3 - very straightforward. */
1111 Assert(pThis->u8ScanSet == 3);
1112 abScan[0] = pKeyDef->makeS3;
1113 abScan[1] = '\0';
1114 if (fKeyDown)
1115 {
1116 RTStrCatP(&pCodes, &cbLeft, (const char *)abScan);
1117 }
1118 else
1119 {
1120 /* Send a key release code unless it's a make only key. */
1121 /// @todo Look up the current typematic setting, not the default!
1122 if (pKeyDef->keyMatic != T_M)
1123 {
1124 RTStrCatP(&pCodes, &cbLeft, "\xF0");
1125 RTStrCatP(&pCodes, &cbLeft, (const char *)abScan);
1126 }
1127 }
1128 /* Feed the bytes to the queue if there is room. */
1129 /// @todo Send overrun code if sequence won't fit?
1130 ps2kInsertStrQueue((GeneriQ *)&pThis->keyQ, abCodes, 0);
1131 }
1132
1133 /* Set up or cancel typematic key repeat. For keyboard usage page only. */
1134 if (u8HidPage == USB_HID_KB_PAGE)
1135 {
1136 if (fKeyDown)
1137 {
1138 if (pThis->u32TypematicKey != u32HidCode)
1139 {
1140 pThis->enmTypematicState = KBD_TMS_DELAY;
1141 pThis->u32TypematicKey = u32HidCode;
1142 TMTimerSetMillies(pThis->CTX_SUFF(pKbdTypematicTimer), pThis->uTypematicDelay);
1143 Log(("Typematic delay %u ms, key %08X\n", pThis->uTypematicDelay, u32HidCode));
1144 }
1145 }
1146 else
1147 {
1148 /* "Typematic operation stops when the last key pressed is released, even
1149 * if other keys are still held down." (IBM PS/2 Tech Ref). The last key pressed
1150 * is the one that's being repeated.
1151 */
1152 if (pThis->u32TypematicKey == u32HidCode)
1153 {
1154 /* This disables the typematic repeat. */
1155 pThis->u32TypematicKey = 0;
1156 pThis->enmTypematicState = KBD_TMS_IDLE;
1157 /* For good measure, we cancel the timer, too. */
1158 TMTimerStop(pThis->CTX_SUFF(pKbdTypematicTimer));
1159 Log(("Typematic action cleared for key %08X\n", u32HidCode));
1160 }
1161 }
1162 }
1163
1164 /* Poke the KBC to update its state. */
1165 KBCUpdateInterrupts(pThis->pParent);
1166
1167 return VINF_SUCCESS;
1168}
1169
1170/* Throttling timer to emulate the finite keyboard communication speed. A PS/2 keyboard is
1171 * limited by the serial link speed and cannot send much more than 1,000 bytes per second.
1172 * Some software (notably Borland Pascal and programs built with its run-time) relies on
1173 * being able to read an incoming scan-code twice. Throttling the data rate enables such
1174 * software to function, while human typists cannot tell any difference.
1175 *
1176 * Note: The throttling is currently only done for keyboard data, not command responses.
1177 * The throttling could and perhaps should be done for any data (including command
1178 * response) scoming from PS/2 devices, both keyboard and auxiliary. That is not currently
1179 * done because it would needlessly slow things down.
1180 */
1181static DECLCALLBACK(void) ps2kThrottleTimer(PPDMDEVINS pDevIns, PTMTIMER pTimer, void *pvUser)
1182{
1183 RT_NOREF2(pDevIns, pTimer);
1184 PPS2K pThis = (PS2K *)pvUser;
1185 unsigned uHaveData;
1186
1187 /* Grab the lock to avoid races with event delivery or EMTs. */
1188 int rc = PDMCritSectEnter(pThis->pCritSectR3, VERR_SEM_BUSY);
1189 AssertReleaseRC(rc);
1190
1191 /* If data is available, poke the KBC. Once the data
1192 * is actually read, the timer may be re-triggered.
1193 */
1194 pThis->fThrottleActive = false;
1195 uHaveData = ps2kInQueue((GeneriQ *)&pThis->keyQ);
1196 LogFlowFunc(("Have%s bytes\n", uHaveData ? "" : " no"));
1197 if (uHaveData)
1198 KBCUpdateInterrupts(pThis->pParent);
1199
1200 PDMCritSectLeave(pThis->pCritSectR3);
1201}
1202
1203/* Timer handler for emulating typematic keys. Note that only the last key
1204 * held down repeats (if typematic).
1205 */
1206static DECLCALLBACK(void) ps2kTypematicTimer(PPDMDEVINS pDevIns, PTMTIMER pTimer, void *pvUser)
1207{
1208 RT_NOREF2(pDevIns, pTimer);
1209 PPS2K pThis = (PS2K *)pvUser;
1210 LogFlowFunc(("Typematic state=%d, key %08X\n", pThis->enmTypematicState, pThis->u32TypematicKey));
1211
1212 /* If the current typematic key is zero, the repeat was canceled just when
1213 * the timer was about to run. In that case, do nothing.
1214 */
1215 if (pThis->u32TypematicKey)
1216 {
1217 if (pThis->enmTypematicState == KBD_TMS_DELAY)
1218 pThis->enmTypematicState = KBD_TMS_REPEAT;
1219
1220 if (pThis->enmTypematicState == KBD_TMS_REPEAT)
1221 {
1222 ps2kProcessKeyEvent(pThis, pThis->u32TypematicKey, true /* Key down */ );
1223 TMTimerSetMillies(pThis->CTX_SUFF(pKbdTypematicTimer), pThis->uTypematicRepeat);
1224 }
1225 }
1226}
1227
1228/* The keyboard BAT is specified to take several hundred milliseconds. We need
1229 * to delay sending the result to the host for at least a tiny little while.
1230 */
1231static DECLCALLBACK(void) ps2kDelayTimer(PPDMDEVINS pDevIns, PTMTIMER pTimer, void *pvUser)
1232{
1233 RT_NOREF2(pDevIns, pTimer);
1234 PPS2K pThis = (PS2K *)pvUser;
1235
1236 LogFlowFunc(("Delay timer: cmd %02X\n", pThis->u8CurrCmd));
1237
1238 AssertMsg(pThis->u8CurrCmd == KCMD_RESET, ("u8CurrCmd=%02x\n", pThis->u8CurrCmd));
1239 ps2kInsertQueue((GeneriQ *)&pThis->cmdQ, KRSP_BAT_OK);
1240 pThis->fScanning = true; /* BAT completion enables scanning! */
1241 pThis->u8CurrCmd = 0;
1242
1243 /// @todo Might want a PS2KCompleteCommand() to push last response, clear command, and kick the KBC...
1244 /* Give the KBC a kick. */
1245 KBCUpdateInterrupts(pThis->pParent);
1246}
1247
1248/* Release any and all currently depressed keys. Used whenever the guest keyboard
1249 * is likely to be out of sync with the host, such as when loading a saved state
1250 * or resuming a suspended host.
1251 */
1252static void ps2kReleaseKeys(PPS2K pThis)
1253{
1254 LogFlowFunc(("Releasing keys...\n"));
1255
1256 for (unsigned uKey = 0; uKey < sizeof(pThis->abDepressedKeys); ++uKey)
1257 if (pThis->abDepressedKeys[uKey])
1258 {
1259 ps2kProcessKeyEvent(pThis, RT_MAKE_U32(USB_HID_KB_PAGE, uKey), false /* key up */);
1260 pThis->abDepressedKeys[uKey] = 0;
1261 }
1262 LogFlowFunc(("Done releasing keys\n"));
1263}
1264
1265
1266/**
1267 * Debug device info handler. Prints basic keyboard state.
1268 *
1269 * @param pDevIns Device instance which registered the info.
1270 * @param pHlp Callback functions for doing output.
1271 * @param pszArgs Argument string. Optional and specific to the handler.
1272 */
1273static DECLCALLBACK(void) ps2kInfoState(PPDMDEVINS pDevIns, PCDBGFINFOHLP pHlp, const char *pszArgs)
1274{
1275 PPS2K pThis = KBDGetPS2KFromDevIns(pDevIns);
1276 NOREF(pszArgs);
1277
1278 pHlp->pfnPrintf(pHlp, "PS/2 Keyboard: scan set %d, scanning %s\n",
1279 pThis->u8ScanSet, pThis->fScanning ? "enabled" : "disabled");
1280 pHlp->pfnPrintf(pHlp, "Active command %02X\n", pThis->u8CurrCmd);
1281 pHlp->pfnPrintf(pHlp, "LED state %02X, Num Lock %s\n", pThis->u8LEDs,
1282 pThis->fNumLockOn ? "on" : "off");
1283 pHlp->pfnPrintf(pHlp, "Typematic delay %ums, repeat period %ums\n",
1284 pThis->uTypematicDelay, pThis->uTypematicRepeat);
1285 pHlp->pfnPrintf(pHlp, "Command queue: %d items (%d max)\n",
1286 pThis->cmdQ.cUsed, pThis->cmdQ.cSize);
1287 pHlp->pfnPrintf(pHlp, "Input queue : %d items (%d max)\n",
1288 pThis->keyQ.cUsed, pThis->keyQ.cSize);
1289 if (pThis->enmTypematicState != KBD_TMS_IDLE)
1290 pHlp->pfnPrintf(pHlp, "Active typematic key %08X (%s)\n", pThis->u32TypematicKey,
1291 pThis->enmTypematicState == KBD_TMS_DELAY ? "delay" : "repeat");
1292}
1293
1294/* -=-=-=-=-=- Keyboard: IBase -=-=-=-=-=- */
1295
1296/**
1297 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
1298 */
1299static DECLCALLBACK(void *) ps2kQueryInterface(PPDMIBASE pInterface, const char *pszIID)
1300{
1301 PPS2K pThis = RT_FROM_MEMBER(pInterface, PS2K, Keyboard.IBase);
1302 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pThis->Keyboard.IBase);
1303 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIKEYBOARDPORT, &pThis->Keyboard.IPort);
1304 return NULL;
1305}
1306
1307
1308/* -=-=-=-=-=- Keyboard: IKeyboardPort -=-=-=-=-=- */
1309
1310/**
1311 * Keyboard event handler.
1312 *
1313 * @returns VBox status code.
1314 * @param pThis The PS2 keyboard instance data.
1315 * @param u32Usage USB HID usage code with key
1316 * press/release flag.
1317 */
1318static int ps2kPutEventWorker(PPS2K pThis, uint32_t u32Usage)
1319{
1320 uint32_t u32HidCode;
1321 uint8_t u8KeyCode;
1322 uint8_t u8HidPage;
1323 bool fKeyDown;
1324 bool fHaveEvent = true;
1325 int rc = VINF_SUCCESS;
1326
1327 /* Extract the usage page and ID and ensure it's valid. */
1328 fKeyDown = !(u32Usage & 0x80000000);
1329 u32HidCode = u32Usage & 0xFFFFFF;
1330 u8HidPage = RT_LOBYTE(RT_HIWORD(u32Usage));
1331 u8KeyCode = RT_LOBYTE(u32Usage);
1332 if (u8HidPage == USB_HID_KB_PAGE)
1333 AssertReturn(u8KeyCode <= VBOX_USB_MAX_USAGE_CODE, VERR_INTERNAL_ERROR);
1334 else
1335 AssertReturn(u8HidPage == USB_HID_CC_PAGE, VERR_INTERNAL_ERROR);
1336
1337 if (fKeyDown)
1338 {
1339 /* Due to host key repeat, we can get key events for keys which are
1340 * already depressed. We need to ignore those. */
1341 if (pThis->abDepressedKeys[u8KeyCode])
1342 fHaveEvent = false;
1343 pThis->abDepressedKeys[u8KeyCode] = 1;
1344 }
1345 else
1346 {
1347 /* NB: We allow key release events for keys which aren't depressed.
1348 * That is unlikely to happen and should not cause trouble.
1349 */
1350 pThis->abDepressedKeys[u8KeyCode] = 0;
1351 }
1352
1353 /* Unless this is a new key press/release, don't even bother. */
1354 if (fHaveEvent)
1355 {
1356 rc = PDMCritSectEnter(pThis->pCritSectR3, VERR_SEM_BUSY);
1357 AssertReleaseRC(rc);
1358
1359 rc = ps2kProcessKeyEvent(pThis, u32HidCode, fKeyDown);
1360
1361 PDMCritSectLeave(pThis->pCritSectR3);
1362 }
1363
1364 return rc;
1365}
1366
1367static DECLCALLBACK(int) ps2kPutEventWrapper(PPDMIKEYBOARDPORT pInterface, uint32_t u32UsageCode)
1368{
1369 PPS2K pThis = RT_FROM_MEMBER(pInterface, PS2K, Keyboard.IPort);
1370 int rc;
1371
1372 LogRelFlowFunc(("key code %08X\n", u32UsageCode));
1373
1374 rc = PDMCritSectEnter(pThis->pCritSectR3, VERR_SEM_BUSY);
1375 AssertReleaseRC(rc);
1376
1377 /* The 'BAT fail' scancode is reused as a signal to release keys. No actual
1378 * key is allowed to use this scancode.
1379 */
1380 if (RT_UNLIKELY(u32UsageCode == KRSP_BAT_FAIL))
1381 {
1382 ps2kReleaseKeys(pThis);
1383 }
1384 else
1385 {
1386 ps2kPutEventWorker(pThis, u32UsageCode);
1387 }
1388
1389 PDMCritSectLeave(pThis->pCritSectR3);
1390
1391 return VINF_SUCCESS;
1392}
1393
1394
1395/**
1396 * Attach command.
1397 *
1398 * This is called to let the device attach to a driver for a
1399 * specified LUN.
1400 *
1401 * This is like plugging in the keyboard after turning on the
1402 * system.
1403 *
1404 * @returns VBox status code.
1405 * @param pThis The PS/2 keyboard instance data.
1406 * @param pDevIns The device instance.
1407 * @param iLUN The logical unit which is being detached.
1408 * @param fFlags Flags, combination of the PDMDEVATT_FLAGS_* \#defines.
1409 */
1410int PS2KAttach(PPS2K pThis, PPDMDEVINS pDevIns, unsigned iLUN, uint32_t fFlags)
1411{
1412 int rc;
1413
1414 /* The LUN must be 0, i.e. keyboard. */
1415 Assert(iLUN == 0);
1416 AssertMsgReturn(fFlags & PDM_TACH_FLAGS_NOT_HOT_PLUG,
1417 ("PS/2 keyboard does not support hotplugging\n"),
1418 VERR_INVALID_PARAMETER);
1419
1420 LogFlowFunc(("iLUN=%d\n", iLUN));
1421
1422 rc = PDMDevHlpDriverAttach(pDevIns, iLUN, &pThis->Keyboard.IBase, &pThis->Keyboard.pDrvBase, "Keyboard Port");
1423 if (RT_SUCCESS(rc))
1424 {
1425 pThis->Keyboard.pDrv = PDMIBASE_QUERY_INTERFACE(pThis->Keyboard.pDrvBase, PDMIKEYBOARDCONNECTOR);
1426 if (!pThis->Keyboard.pDrv)
1427 {
1428 AssertLogRelMsgFailed(("LUN #0 doesn't have a keyboard interface! rc=%Rrc\n", rc));
1429 rc = VERR_PDM_MISSING_INTERFACE;
1430 }
1431 }
1432 else if (rc == VERR_PDM_NO_ATTACHED_DRIVER)
1433 {
1434 Log(("%s/%d: warning: no driver attached to LUN #0!\n", pDevIns->pReg->szName, pDevIns->iInstance));
1435 rc = VINF_SUCCESS;
1436 }
1437 else
1438 AssertLogRelMsgFailed(("Failed to attach LUN #0! rc=%Rrc\n", rc));
1439
1440 return rc;
1441}
1442
1443void PS2KSaveState(PPS2K pThis, PSSMHANDLE pSSM)
1444{
1445 uint32_t cPressed = 0;
1446 uint32_t cbTMSSize = 0;
1447
1448 LogFlowFunc(("Saving PS2K state\n"));
1449
1450 /* Save the basic keyboard state. */
1451 SSMR3PutU8(pSSM, pThis->u8CurrCmd);
1452 SSMR3PutU8(pSSM, pThis->u8LEDs);
1453 SSMR3PutU8(pSSM, pThis->u8TypematicCfg);
1454 SSMR3PutU8(pSSM, (uint8_t)pThis->u32TypematicKey);
1455 SSMR3PutU8(pSSM, pThis->u8Modifiers);
1456 SSMR3PutU8(pSSM, pThis->u8ScanSet);
1457 SSMR3PutU8(pSSM, pThis->enmTypematicState);
1458 SSMR3PutBool(pSSM, pThis->fNumLockOn);
1459 SSMR3PutBool(pSSM, pThis->fScanning);
1460
1461 /* Save the command and keystroke queues. */
1462 ps2kSaveQueue(pSSM, (GeneriQ *)&pThis->cmdQ);
1463 ps2kSaveQueue(pSSM, (GeneriQ *)&pThis->keyQ);
1464
1465 /* Save the command delay timer. Note that the typematic repeat
1466 * timer is *not* saved.
1467 */
1468 TMR3TimerSave(pThis->CTX_SUFF(pKbdDelayTimer), pSSM);
1469
1470 /* Save any pressed keys. This is necessary to avoid "stuck"
1471 * keys after a restore. Needs two passes.
1472 */
1473 for (unsigned i = 0; i < sizeof(pThis->abDepressedKeys); ++i)
1474 if (pThis->abDepressedKeys[i])
1475 ++cPressed;
1476
1477 SSMR3PutU32(pSSM, cPressed);
1478
1479 for (unsigned uKey = 0; uKey < sizeof(pThis->abDepressedKeys); ++uKey)
1480 if (pThis->abDepressedKeys[uKey])
1481 SSMR3PutU8(pSSM, uKey);
1482
1483 /* Save the typematic settings for Scan Set 3. */
1484 SSMR3PutU32(pSSM, cbTMSSize);
1485 /* Currently not implemented. */
1486}
1487
1488int PS2KLoadState(PPS2K pThis, PSSMHANDLE pSSM, uint32_t uVersion)
1489{
1490 uint8_t u8;
1491 uint32_t cPressed;
1492 uint32_t cbTMSSize;
1493 int rc;
1494
1495 NOREF(uVersion);
1496 LogFlowFunc(("Loading PS2K state version %u\n", uVersion));
1497
1498 /* Load the basic keyboard state. */
1499 SSMR3GetU8(pSSM, &pThis->u8CurrCmd);
1500 SSMR3GetU8(pSSM, &pThis->u8LEDs);
1501 SSMR3GetU8(pSSM, &pThis->u8TypematicCfg);
1502 SSMR3GetU8(pSSM, &u8);
1503 /* Reconstruct the 32-bit code from the 8-bit value in saved state. */
1504 pThis->u32TypematicKey = u8 ? RT_MAKE_U32(USB_HID_KB_PAGE, u8) : 0;
1505 SSMR3GetU8(pSSM, &pThis->u8Modifiers);
1506 SSMR3GetU8(pSSM, &pThis->u8ScanSet);
1507 SSMR3GetU8(pSSM, &u8);
1508 pThis->enmTypematicState = (tmatic_state_t)u8;
1509 SSMR3GetBool(pSSM, &pThis->fNumLockOn);
1510 SSMR3GetBool(pSSM, &pThis->fScanning);
1511
1512 /* Load the command and keystroke queues. */
1513 rc = ps2kLoadQueue(pSSM, (GeneriQ *)&pThis->cmdQ);
1514 AssertRCReturn(rc, rc);
1515 rc = ps2kLoadQueue(pSSM, (GeneriQ *)&pThis->keyQ);
1516 AssertRCReturn(rc, rc);
1517
1518 /* Load the command delay timer, just in case. */
1519 rc = TMR3TimerLoad(pThis->CTX_SUFF(pKbdDelayTimer), pSSM);
1520 AssertRCReturn(rc, rc);
1521
1522 /* Recalculate the typematic delay/rate. */
1523 ps2kSetupTypematic(pThis, pThis->u8TypematicCfg);
1524
1525 /* Fake key up events for keys that were held down at the time the state was saved. */
1526 rc = SSMR3GetU32(pSSM, &cPressed);
1527 AssertRCReturn(rc, rc);
1528
1529 /* If any keys were down, load and then release them. */
1530 if (cPressed)
1531 {
1532 for (unsigned i = 0; i < cPressed; ++i)
1533 {
1534 rc = SSMR3GetU8(pSSM, &u8);
1535 AssertRCReturn(rc, rc);
1536 pThis->abDepressedKeys[u8] = 1;
1537 }
1538 }
1539
1540 /* Load typematic settings for Scan Set 3. */
1541 rc = SSMR3GetU32(pSSM, &cbTMSSize);
1542 AssertRCReturn(rc, rc);
1543
1544 while (cbTMSSize--)
1545 {
1546 rc = SSMR3GetU8(pSSM, &u8);
1547 AssertRCReturn(rc, rc);
1548 }
1549
1550 return rc;
1551}
1552
1553int PS2KLoadDone(PPS2K pThis, PSSMHANDLE pSSM)
1554{
1555 RT_NOREF1(pSSM);
1556
1557 /* This *must* be done after the inital load because it may trigger
1558 * interrupts and change the interrupt controller state.
1559 */
1560 ps2kReleaseKeys(pThis);
1561 ps2kNotifyLedsState(pThis, pThis->u8LEDs);
1562 return VINF_SUCCESS;
1563}
1564
1565void PS2KReset(PPS2K pThis)
1566{
1567 LogFlowFunc(("Resetting PS2K\n"));
1568
1569 pThis->fScanning = true;
1570 pThis->fThrottleActive = false;
1571 pThis->u8ScanSet = 2;
1572 pThis->u8CurrCmd = 0;
1573 pThis->u8Modifiers = 0;
1574 pThis->u32TypematicKey = 0;
1575 pThis->enmTypematicState = KBD_TMS_IDLE;
1576
1577 /* Clear queues and any pressed keys. */
1578 memset(pThis->abDepressedKeys, 0, sizeof(pThis->abDepressedKeys));
1579 ps2kClearQueue((GeneriQ *)&pThis->cmdQ);
1580 ps2kSetDefaults(pThis); /* Also clears keystroke queue. */
1581
1582 /* Activate the PS/2 keyboard by default. */
1583 if (pThis->Keyboard.pDrv)
1584 pThis->Keyboard.pDrv->pfnSetActive(pThis->Keyboard.pDrv, true);
1585}
1586
1587void PS2KRelocate(PPS2K pThis, RTGCINTPTR offDelta, PPDMDEVINS pDevIns)
1588{
1589 RT_NOREF1(pDevIns);
1590 LogFlowFunc(("Relocating PS2K\n"));
1591 pThis->pKbdDelayTimerRC = TMTimerRCPtr(pThis->pKbdDelayTimerR3);
1592 pThis->pKbdTypematicTimerRC = TMTimerRCPtr(pThis->pKbdTypematicTimerR3);
1593 pThis->pThrottleTimerRC = TMTimerRCPtr(pThis->pThrottleTimerR3);
1594 NOREF(offDelta);
1595}
1596
1597int PS2KConstruct(PPS2K pThis, PPDMDEVINS pDevIns, void *pParent, int iInstance, PCFGMNODE pCfg)
1598{
1599 RT_NOREF2(pDevIns, iInstance);
1600 LogFlowFunc(("iInstance=%d\n", iInstance));
1601
1602 pThis->pParent = pParent;
1603
1604 bool fThrottleEnabled;
1605 int rc = CFGMR3QueryBoolDef(pCfg, "KbdThrottleEnabled", &fThrottleEnabled, true);
1606 if (RT_FAILURE(rc))
1607 return PDMDEV_SET_ERROR(pDevIns, rc, N_("Failed to query \"KbdThrottleEnabled\" from the config"));
1608 Log(("KbdThrottleEnabled=%u\n", fThrottleEnabled));
1609 pThis->fThrottleEnabled = fThrottleEnabled;
1610
1611 /* Initialize the queues. */
1612 pThis->keyQ.cSize = KBD_KEY_QUEUE_SIZE;
1613 pThis->cmdQ.cSize = KBD_CMD_QUEUE_SIZE;
1614
1615 pThis->Keyboard.IBase.pfnQueryInterface = ps2kQueryInterface;
1616 pThis->Keyboard.IPort.pfnPutEventHid = ps2kPutEventWrapper;
1617
1618 /*
1619 * Initialize the critical section pointer(s).
1620 */
1621 pThis->pCritSectR3 = pDevIns->pCritSectRoR3;
1622
1623 /*
1624 * Create the input rate throttling timer.
1625 */
1626 PTMTIMER pTimer;
1627 rc = PDMDevHlpTMTimerCreate(pDevIns, TMCLOCK_VIRTUAL, ps2kThrottleTimer, pThis,
1628 TMTIMER_FLAGS_DEFAULT_CRIT_SECT, "PS2K Throttle Timer", &pTimer);
1629 if (RT_FAILURE(rc))
1630 return rc;
1631
1632 pThis->pThrottleTimerR3 = pTimer;
1633 pThis->pThrottleTimerR0 = TMTimerR0Ptr(pTimer);
1634 pThis->pThrottleTimerRC = TMTimerRCPtr(pTimer);
1635
1636 /*
1637 * Create the typematic delay/repeat timer.
1638 */
1639 rc = PDMDevHlpTMTimerCreate(pDevIns, TMCLOCK_VIRTUAL, ps2kTypematicTimer, pThis,
1640 TMTIMER_FLAGS_DEFAULT_CRIT_SECT, "PS2K Typematic Timer", &pTimer);
1641 if (RT_FAILURE(rc))
1642 return rc;
1643
1644 pThis->pKbdTypematicTimerR3 = pTimer;
1645 pThis->pKbdTypematicTimerR0 = TMTimerR0Ptr(pTimer);
1646 pThis->pKbdTypematicTimerRC = TMTimerRCPtr(pTimer);
1647
1648 /*
1649 * Create the command delay timer.
1650 */
1651 rc = PDMDevHlpTMTimerCreate(pDevIns, TMCLOCK_VIRTUAL, ps2kDelayTimer, pThis,
1652 TMTIMER_FLAGS_DEFAULT_CRIT_SECT, "PS2K Delay Timer", &pTimer);
1653 if (RT_FAILURE(rc))
1654 return rc;
1655
1656 pThis->pKbdDelayTimerR3 = pTimer;
1657 pThis->pKbdDelayTimerR0 = TMTimerR0Ptr(pTimer);
1658 pThis->pKbdDelayTimerRC = TMTimerRCPtr(pTimer);
1659
1660 /*
1661 * Register debugger info callbacks.
1662 */
1663 PDMDevHlpDBGFInfoRegister(pDevIns, "ps2k", "Display PS/2 keyboard state.", ps2kInfoState);
1664
1665 return rc;
1666}
1667
1668#endif
1669
1670/// @todo The following should live with the KBC implementation.
1671
1672/* Table used by the keyboard controller to optionally translate the incoming
1673 * keyboard data. Note that the translation is designed for essentially taking
1674 * Scan Set 2 input and producing Scan Set 1 output, but can be turned on and
1675 * off regardless of what the keyboard is sending.
1676 */
1677static uint8_t aAT2PC[128] = {
1678 0xff,0x43,0x41,0x3f,0x3d,0x3b,0x3c,0x58,0x64,0x44,0x42,0x40,0x3e,0x0f,0x29,0x59,
1679 0x65,0x38,0x2a,0x70,0x1d,0x10,0x02,0x5a,0x66,0x71,0x2c,0x1f,0x1e,0x11,0x03,0x5b,
1680 0x67,0x2e,0x2d,0x20,0x12,0x05,0x04,0x5c,0x68,0x39,0x2f,0x21,0x14,0x13,0x06,0x5d,
1681 0x69,0x31,0x30,0x23,0x22,0x15,0x07,0x5e,0x6a,0x72,0x32,0x24,0x16,0x08,0x09,0x5f,
1682 0x6b,0x33,0x25,0x17,0x18,0x0b,0x0a,0x60,0x6c,0x34,0x35,0x26,0x27,0x19,0x0c,0x61,
1683 0x6d,0x73,0x28,0x74,0x1a,0x0d,0x62,0x6e,0x3a,0x36,0x1c,0x1b,0x75,0x2b,0x63,0x76,
1684 0x55,0x56,0x77,0x78,0x79,0x7a,0x0e,0x7b,0x7c,0x4f,0x7d,0x4b,0x47,0x7e,0x7f,0x6f,
1685 0x52,0x53,0x50,0x4c,0x4d,0x48,0x01,0x45,0x57,0x4e,0x51,0x4a,0x37,0x49,0x46,0x54
1686};
1687
1688/**
1689 * Convert an AT (Scan Set 2) scancode to PC (Scan Set 1).
1690 *
1691 * @param state Current state of the translator
1692 * (xlat_state_t).
1693 * @param scanIn Incoming scan code.
1694 * @param pScanOut Pointer to outgoing scan code. The
1695 * contents are only valid if returned
1696 * state is not XS_BREAK.
1697 *
1698 * @return xlat_state_t New state of the translator.
1699 */
1700int32_t XlateAT2PC(int32_t state, uint8_t scanIn, uint8_t *pScanOut)
1701{
1702 uint8_t scan_in;
1703 uint8_t scan_out;
1704
1705 Assert(pScanOut);
1706 Assert(state == XS_IDLE || state == XS_BREAK || state == XS_HIBIT);
1707
1708 /* Preprocess the scan code for a 128-entry translation table. */
1709 if (scanIn == 0x83) /* Check for F7 key. */
1710 scan_in = 0x02;
1711 else if (scanIn == 0x84) /* Check for SysRq key. */
1712 scan_in = 0x7f;
1713 else
1714 scan_in = scanIn;
1715
1716 /* Values 0x80 and above are passed through, except for 0xF0
1717 * which indicates a key release.
1718 */
1719 if (scan_in < 0x80)
1720 {
1721 scan_out = aAT2PC[scan_in];
1722 /* Turn into break code if required. */
1723 if (state == XS_BREAK || state == XS_HIBIT)
1724 scan_out |= 0x80;
1725
1726 state = XS_IDLE;
1727 }
1728 else
1729 {
1730 /* NB: F0 E0 10 will be translated to E0 E5 (high bit set on last byte)! */
1731 if (scan_in == 0xF0) /* Check for break code. */
1732 state = XS_BREAK;
1733 else if (state == XS_BREAK)
1734 state = XS_HIBIT; /* Remember the break bit. */
1735 scan_out = scan_in;
1736 }
1737 LogFlowFunc(("scan code %02X translated to %02X; new state is %d\n",
1738 scanIn, scan_out, state));
1739
1740 *pScanOut = scan_out;
1741 return state;
1742}
1743
1744#endif /* !VBOX_DEVICE_STRUCT_TESTCASE */
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