VirtualBox

source: vbox/trunk/src/VBox/Additions/WINNT/Mouse/NT4/VBoxPS2NT.cpp@ 40353

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

Additions/WINNT/Mouse/NT4: fix OSE

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 113.7 KB
Line 
1/* $Id: VBoxPS2NT.cpp 40353 2012-03-05 13:18:23Z vboxsync $ */
2/** @file
3 * VBox NT4 Mouse Driver
4 */
5
6/*
7 * Copyright (C) 2011-2012 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#define LOG_GROUP LOG_GROUP_DRV_MOUSE
19#include <iprt/asm.h>
20#include <VBox/err.h>
21#include <VBox/log.h>
22#include <VBox/VBoxGuestLib.h>
23
24#include <stdarg.h>
25#include <string.h>
26RT_C_DECLS_BEGIN
27#undef PAGE_SIZE
28#undef PAGE_SHIFT
29#include <ntddk.h>
30#include <ntddkbd.h>
31#include <ntddmou.h>
32RT_C_DECLS_END
33
34/* not available on NT4 */
35#undef ExFreePool
36#undef ExAllocatePool
37
38/* i8042 mouse status bits */
39#define LEFT_BUTTON_DOWN 0x01
40#define RIGHT_BUTTON_DOWN 0x02
41#define MIDDLE_BUTTON_DOWN 0x04
42#define X_DATA_SIGN 0x10
43#define Y_DATA_SIGN 0x20
44#define X_OVERFLOW 0x40
45#define Y_OVERFLOW 0x80
46
47#define MOUSE_SIGN_OVERFLOW_MASK (X_DATA_SIGN | Y_DATA_SIGN | X_OVERFLOW | Y_OVERFLOW)
48
49#define MOUSE_MAXIMUM_POSITIVE_DELTA 0x000000FF
50#define MOUSE_MAXIMUM_NEGATIVE_DELTA 0xFFFFFF00
51
52#define KEYBOARD_HARDWARE_PRESENT 0x01
53#define MOUSE_HARDWARE_PRESENT 0x02
54#define BALLPOINT_HARDWARE_PRESENT 0x04
55#define WHEELMOUSE_HARDWARE_PRESENT 0x08
56
57#define I8X_PUT_COMMAND_BYTE(Address, Byte) WRITE_PORT_UCHAR(Address, (UCHAR) Byte)
58#define I8X_PUT_DATA_BYTE(Address, Byte) WRITE_PORT_UCHAR(Address, (UCHAR) Byte)
59#define I8X_GET_STATUS_BYTE(Address) READ_PORT_UCHAR(Address)
60#define I8X_GET_DATA_BYTE(Address) READ_PORT_UCHAR(Address)
61
62/* commands to the i8042 controller */
63#define I8042_READ_CONTROLLER_COMMAND_BYTE 0x20
64#define I8042_WRITE_CONTROLLER_COMMAND_BYTE 0x60
65#define I8042_DISABLE_MOUSE_DEVICE 0xA7
66#define I8042_ENABLE_MOUSE_DEVICE 0xA8
67#define I8042_AUXILIARY_DEVICE_TEST 0xA9
68#define I8042_KEYBOARD_DEVICE_TEST 0xAB
69#define I8042_DISABLE_KEYBOARD_DEVICE 0xAD
70#define I8042_ENABLE_KEYBOARD_DEVICE 0xAE
71#define I8042_WRITE_TO_AUXILIARY_DEVICE 0xD4
72
73/* i8042 Controller Command Byte */
74#define CCB_ENABLE_KEYBOARD_INTERRUPT 0x01
75#define CCB_ENABLE_MOUSE_INTERRUPT 0x02
76#define CCB_DISABLE_KEYBOARD_DEVICE 0x10
77#define CCB_DISABLE_MOUSE_DEVICE 0x20
78#define CCB_KEYBOARD_TRANSLATE_MODE 0x40
79
80/* i8042 Controller Status Register bits */
81#define OUTPUT_BUFFER_FULL 0x01
82#define INPUT_BUFFER_FULL 0x02
83#define MOUSE_OUTPUT_BUFFER_FULL 0x20
84
85/* i8042 responses */
86#define ACKNOWLEDGE 0xFA
87#define RESEND 0xFE
88
89/* commands to the keyboard (through the 8042 data port) */
90#define SET_KEYBOARD_INDICATORS 0xED
91#define SELECT_SCAN_CODE_SET 0xF0
92#define READ_KEYBOARD_ID 0xF2
93#define SET_KEYBOARD_TYPEMATIC 0xF3
94#define SET_ALL_TYPEMATIC_MAKE_BREAK 0xFA
95#define KEYBOARD_RESET 0xFF
96
97/* commands to the mouse (through the 8042 data port) */
98#define SET_MOUSE_SCALING_1TO1 0xE6
99#define SET_MOUSE_RESOLUTION 0xE8
100#define READ_MOUSE_STATUS 0xE9
101#define GET_DEVICE_ID 0xF2
102#define SET_MOUSE_SAMPLING_RATE 0xF3
103#define ENABLE_MOUSE_TRANSMISSION 0xF4
104#define MOUSE_RESET 0xFF
105
106/* mouse responses */
107#define MOUSE_COMPLETE 0xAA
108#define MOUSE_ID_BYTE 0x00
109#define WHEELMOUSE_ID_BYTE 0x03
110
111/* maximum number of pointer/keyboard port names the port driver */
112#define POINTER_PORTS_MAXIMUM 8
113#define KEYBOARD_PORTS_MAXIMUM 8
114
115/* NtDeviceIoControlFile internal IoControlCode values for keyboard device */
116#define IOCTL_INTERNAL_KEYBOARD_CONNECT CTL_CODE(FILE_DEVICE_KEYBOARD, 0x0080, METHOD_NEITHER, FILE_ANY_ACCESS)
117#define IOCTL_INTERNAL_KEYBOARD_DISCONNECT CTL_CODE(FILE_DEVICE_KEYBOARD, 0x0100, METHOD_NEITHER, FILE_ANY_ACCESS)
118#define IOCTL_INTERNAL_KEYBOARD_ENABLE CTL_CODE(FILE_DEVICE_KEYBOARD, 0x0200, METHOD_NEITHER, FILE_ANY_ACCESS)
119#define IOCTL_INTERNAL_KEYBOARD_DISABLE CTL_CODE(FILE_DEVICE_KEYBOARD, 0x0400, METHOD_NEITHER, FILE_ANY_ACCESS)
120
121/* NtDeviceIoControlFile internal IoControlCode values for mouse device */
122#define IOCTL_INTERNAL_MOUSE_CONNECT CTL_CODE(FILE_DEVICE_MOUSE, 0x0080, METHOD_NEITHER, FILE_ANY_ACCESS)
123#define IOCTL_INTERNAL_MOUSE_DISCONNECT CTL_CODE(FILE_DEVICE_MOUSE, 0x0100, METHOD_NEITHER, FILE_ANY_ACCESS)
124#define IOCTL_INTERNAL_MOUSE_ENABLE CTL_CODE(FILE_DEVICE_MOUSE, 0x0200, METHOD_NEITHER, FILE_ANY_ACCESS)
125#define IOCTL_INTERNAL_MOUSE_DISABLE CTL_CODE(FILE_DEVICE_MOUSE, 0x0400, METHOD_NEITHER, FILE_ANY_ACCESS)
126
127/* i8042 controller input/output ports */
128typedef enum _I8042IOPORTTYPE
129{
130 i8042Dat = 0,
131 i8042Cmd,
132 i8042MaxPorts
133} I8042IOPORTTYPE;
134
135/* device types attached to the i8042 controller */
136typedef enum _I8042DEVTYPE
137{
138 CtrlDevType,
139 KbdDevType,
140 MouDevType,
141 NoDevice
142} I8042DEVTYPE;
143
144/* keyboard output states */
145typedef enum _KBDSTATE
146{
147 Idle,
148 SendFirstByte,
149 SendLastByte
150} KBDSTATE;
151
152/* keyboard scan code input states */
153typedef enum _KBDSCANSTATE
154{
155 Normal,
156 GotE0,
157 GotE1
158} KBDSCANSTATE;
159
160/* mouse states */
161typedef enum _MOUSTATE
162{
163 MouseIdle,
164 XMovement,
165 YMovement,
166 ZMovement,
167 MouseExpectingACK
168} MOUSTATE;
169
170typedef VOID (*PSERVICECALLBACK)(PVOID Ctx, PVOID pArg1, PVOID pArg2, PVOID pArg3);
171
172typedef struct _CONNECT_DATA
173{
174 PDEVICE_OBJECT ClassDeviceObject;
175 PSERVICECALLBACK ClassService;
176} CONNECT_DATA, *PCONNECT_DATA;
177
178typedef struct _KBDSETPACKET
179{
180 USHORT State;
181 UCHAR FirstByte;
182 UCHAR LastByte;
183} KBDSETPACKET, *PKBDSETPACKET;
184
185typedef struct _I8042CFGINF
186{
187 INTERFACE_TYPE InterfaceType; /**< bus interface type */
188 ULONG uBusNr; /**< bus number */
189 ULONG cPorts;
190 CM_PARTIAL_RESOURCE_DESCRIPTOR aPorts[i8042MaxPorts];
191 CM_PARTIAL_RESOURCE_DESCRIPTOR KbdInt;
192 CM_PARTIAL_RESOURCE_DESCRIPTOR MouInt;
193 BOOLEAN fFloatSave; /**< weather to save floating point context */
194 USHORT iResend; /**< number of retries allowed */
195 USHORT PollingIterations; /**< number of polling iterations */
196 USHORT PollingIterationsMaximum;
197 USHORT PollStatusIterations;
198 USHORT StallMicroseconds;
199 KEYBOARD_ATTRIBUTES KbdAttr;
200 KEYBOARD_TYPEMATIC_PARAMETERS KeyRepeatCurrent;
201 KEYBOARD_INDICATOR_PARAMETERS KbdInd;
202 MOUSE_ATTRIBUTES MouAttr;
203 USHORT MouseResolution;
204 ULONG EnableWheelDetection;
205} I8042CFGINF, *PI8042CFGINF;
206
207typedef struct _PORTKBDEXT
208{
209 CONNECT_DATA ConnectData;
210 ULONG cInput;
211 PKEYBOARD_INPUT_DATA InputData;
212 PKEYBOARD_INPUT_DATA DataIn;
213 PKEYBOARD_INPUT_DATA DataOut;
214 PKEYBOARD_INPUT_DATA DataEnd;
215 KEYBOARD_INPUT_DATA CurrentInput;
216 KBDSCANSTATE CurrentScanState;
217 KBDSETPACKET CurrentOutput;
218 USHORT ResendCount;
219 KTIMER DataConsumptionTimer;
220 USHORT UnitId;
221} PORTKBDEXT, *PPORTKBDEXT;
222
223typedef struct _PORTMOUEXT
224{
225 CONNECT_DATA ConnectData;
226 ULONG cInput;
227 PMOUSE_INPUT_DATA InputData;
228 PMOUSE_INPUT_DATA DataIn;
229 PMOUSE_INPUT_DATA DataOut;
230 PMOUSE_INPUT_DATA DataEnd;
231 MOUSE_INPUT_DATA CurrentInput;
232 USHORT InputState;
233 UCHAR uCurrSignAndOverflow;
234 UCHAR uPrevSignAndOverflow;
235 UCHAR PreviousButtons;
236 KTIMER DataConsumptionTimer;
237 LARGE_INTEGER PreviousTick;
238 USHORT UnitId;
239 ULONG SynchTickCount;
240 UCHAR LastByteReceived;
241} PORTMOUEXT, *PPORTMOUEXT;
242
243typedef struct _DEVEXT
244{
245 ULONG HardwarePresent;
246 volatile uint32_t KeyboardEnableCount;
247 volatile uint32_t MouseEnableCount;
248 PDEVICE_OBJECT pDevObj;
249 PUCHAR DevRegs[i8042MaxPorts];
250 PORTKBDEXT KbdExt;
251 PORTMOUEXT MouExt;
252 I8042CFGINF Cfg;
253 PKINTERRUPT KbdIntObj;
254 PKINTERRUPT MouIntObj;
255 KSPIN_LOCK ShIntObj;
256 KDPC RetriesExceededDpc;
257 KDPC KeyboardIsrDpc;
258 KDPC KeyboardIsrDpcRetry;
259 LONG DpcInterlockKeyboard;
260 KDPC MouseIsrDpc;
261 KDPC MouseIsrDpcRetry;
262 LONG DpcInterlockMouse;
263 KDPC TimeOutDpc;
264 KTIMER CommandTimer;
265 LONG TimerCount;
266 BOOLEAN fUnmapRegs;
267 VMMDevReqMouseStatus *pReq;
268} DEVEXT, *PDEVEXT;
269
270typedef struct _INITEXT
271{
272 DEVEXT DevExt;
273} INITEXT, *PINITEXT;
274
275typedef struct _I8042INITDATACTX
276{
277 PDEVEXT pDevExt;
278 int DevType;
279} I8042INITDATACTX, *PI8042INITDATACTX;
280
281typedef struct _I8042TRANSMITCCBCTX
282{
283 ULONG HwDisEnMask;
284 BOOLEAN fAndOp;
285 UCHAR ByteMask;
286 NTSTATUS Status;
287} I8042TRANSMITCCBCTX, *PI8042TRANSMITCCBCTX;
288
289typedef struct _GETDATAPTRCTX
290{
291 PDEVEXT pDevExt;
292 int DevType;
293 PVOID DataIn;
294 PVOID DataOut;
295 ULONG cInput;
296} GETDATAPTRCTX, *PGETDATAPTRCTX;
297
298typedef struct _SETDATAPTRCTX
299{
300 PDEVEXT pDevExt;
301 int DevType;
302 ULONG cInput;
303 PVOID DataOut;
304} SETDATAPTRCTX, *PSETDATAPTRCTX;
305
306typedef struct _TIMERCTX
307{
308 PDEVICE_OBJECT pDevObj;
309 PLONG TimerCounter;
310 LONG NewTimerCount;
311} TIMERCTX, *PTIMERCTX;
312
313typedef struct _KBDINITIATECTX
314{
315 PDEVICE_OBJECT pDevObj;
316 UCHAR FirstByte;
317 UCHAR LastByte;
318} KBDINITIATECTX, *PKBDINITIATECTX;
319
320typedef enum _OPTYPE
321{
322 IncrementOperation,
323 DecrementOperation,
324 WriteOperation,
325} OPTYPE;
326
327typedef struct _VAROPCTX
328{
329 PLONG VariableAddress;
330 OPTYPE Operation;
331 PLONG NewValue;
332} VAROPCTX, *PVAROPCTX;
333
334typedef struct _KBDTYPEINFO
335{
336 USHORT cFunctionKeys;
337 USHORT cIndicators;
338 USHORT cKeysTotal;
339} KBDTYPEINFO;
340
341static const INDICATOR_LIST s_aIndicators[3] =
342{
343 {0x3A, KEYBOARD_CAPS_LOCK_ON},
344 {0x45, KEYBOARD_NUM_LOCK_ON},
345 {0x46, KEYBOARD_SCROLL_LOCK_ON}
346};
347
348static const KBDTYPEINFO s_aKeybType[4] =
349{
350 {10, 3, 84}, /* PC/XT 83- 84-key keyboard (and compatibles) */
351 {12, 3, 102}, /* Olivetti M24 102-key keyboard (and compatibles) */
352 {10, 3, 84}, /* All AT type keyboards (84-86 keys) */
353 {12, 3, 101} /* Enhanced 101- or 102-key keyboards (and compatibles) */
354};
355
356RT_C_DECLS_BEGIN
357static NTSTATUS MouFindWheel(PDEVICE_OBJECT pDevObj);
358static VOID MouGetRegstry(PINITEXT pInit, PUNICODE_STRING RegistryPath,
359 PUNICODE_STRING KeyboardDeviceName, PUNICODE_STRING PointerDeviceName);
360static NTSTATUS MouInitHw(PDEVICE_OBJECT pDevObj);
361static VOID KbdGetRegstry(PINITEXT pInit, PUNICODE_STRING RegistryPath,
362 PUNICODE_STRING KeyboardDeviceName, PUNICODE_STRING PointerDeviceName);
363static NTSTATUS KbdInitHw(PDEVICE_OBJECT pDevObj);
364static VOID HwGetRegstry(PINITEXT pInit, PUNICODE_STRING RegistryPath,
365 PUNICODE_STRING KeyboardDeviceName, PUNICODE_STRING PointerDeviceName);
366static VOID CreateResList(PDEVEXT pDevExt, PCM_RESOURCE_LIST *pResList, PULONG pResListSize);
367static VOID InitHw(PDEVICE_OBJECT pDevObj);
368static NTSTATUS MouCallOut(PVOID pCtx, PUNICODE_STRING PathName,
369 INTERFACE_TYPE BusType, ULONG uBusNr, PKEY_VALUE_FULL_INFORMATION *pBusInf,
370 CONFIGURATION_TYPE uCtrlType, ULONG uCtrlNr, PKEY_VALUE_FULL_INFORMATION *pCtrlInf,
371 CONFIGURATION_TYPE uPrfType, ULONG uPrfNr, PKEY_VALUE_FULL_INFORMATION *pPrfInf);
372static NTSTATUS KbdCallOut(PVOID pCtx, PUNICODE_STRING PathName,
373 INTERFACE_TYPE BusType, ULONG uBusNr, PKEY_VALUE_FULL_INFORMATION *pBusInf,
374 CONFIGURATION_TYPE uCtrlType, ULONG uCtrlNr, PKEY_VALUE_FULL_INFORMATION *pCtrlInf,
375 CONFIGURATION_TYPE uPrfType, ULONG uPrfNr, PKEY_VALUE_FULL_INFORMATION *pPrfInf);
376/* */ NTSTATUS DriverEntry(PDRIVER_OBJECT pDrvObj, PUNICODE_STRING RegistryPath);
377RT_C_DECLS_END
378
379#ifdef ALLOC_PRAGMA
380#pragma alloc_text(INIT,CreateResList)
381#pragma alloc_text(INIT,MouFindWheel)
382#pragma alloc_text(INIT,KbdInitHw)
383#pragma alloc_text(INIT,KbdGetRegstry)
384#pragma alloc_text(INIT,KbdCallOut)
385#pragma alloc_text(INIT,MouInitHw)
386#pragma alloc_text(INIT,MouGetRegstry)
387#pragma alloc_text(INIT,MouCallOut)
388#pragma alloc_text(INIT,InitHw)
389#pragma alloc_text(INIT,HwGetRegstry)
390#pragma alloc_text(INIT,DriverEntry)
391#endif
392
393static BOOLEAN MouDataToQueue(PPORTMOUEXT MouExt, PMOUSE_INPUT_DATA InputData)
394{
395 if ( MouExt->DataIn == MouExt->DataOut
396 && MouExt->cInput)
397 return FALSE;
398
399 *(MouExt->DataIn) = *InputData;
400 MouExt->cInput++;
401 MouExt->DataIn++;
402 if (MouExt->DataIn == MouExt->DataEnd)
403 MouExt->DataIn = MouExt->InputData;
404 return TRUE;
405}
406
407static BOOLEAN KbdDataToQueue(PPORTKBDEXT KbdExt, PKEYBOARD_INPUT_DATA InputData)
408{
409 PKEYBOARD_INPUT_DATA previousDataIn;
410
411 if ( KbdExt->DataIn == KbdExt->DataOut
412 && KbdExt->cInput)
413 {
414 if (KbdExt->DataIn == KbdExt->InputData)
415 previousDataIn = KbdExt->DataEnd;
416 else
417 previousDataIn = KbdExt->DataIn - 1;
418 previousDataIn->MakeCode = KEYBOARD_OVERRUN_MAKE_CODE;
419 previousDataIn->Flags = 0;
420 return FALSE;
421 }
422
423 *(KbdExt->DataIn) = *InputData;
424 KbdExt->cInput++;
425 KbdExt->DataIn++;
426 if (KbdExt->DataIn == KbdExt->DataEnd)
427 KbdExt->DataIn = KbdExt->InputData;
428 return TRUE;
429}
430
431/**
432 * Queues the current input data to be processed by a DPC outside the ISR
433 */
434static VOID QueueInput(PDEVICE_OBJECT pDevObj)
435{
436 PDEVEXT pDevExt = (PDEVEXT)pDevObj->DeviceExtension;
437 if (pDevExt->MouseEnableCount)
438 {
439 pDevExt->MouExt.CurrentInput.UnitId = pDevExt->MouExt.UnitId;
440 if (!MouDataToQueue(&pDevExt->MouExt, &pDevExt->MouExt.CurrentInput))
441 {
442 }
443 else if (pDevExt->DpcInterlockMouse >= 0)
444 pDevExt->DpcInterlockMouse++;
445 else
446 KeInsertQueueDpc(&pDevExt->MouseIsrDpc, pDevObj->CurrentIrp, NULL);
447 }
448}
449
450/**
451 * Drain the i8042 controller buffer.
452 */
453static VOID DrainOutBuf(PUCHAR DataAddress, PUCHAR CommandAddress)
454{
455 UCHAR byte;
456 for (unsigned i = 0; i < 2000; i++)
457 {
458 if (!(I8X_GET_STATUS_BYTE(CommandAddress) & INPUT_BUFFER_FULL))
459 break;
460 KeStallExecutionProcessor(500);
461 }
462 while (I8X_GET_STATUS_BYTE(CommandAddress) & OUTPUT_BUFFER_FULL)
463 byte = I8X_GET_DATA_BYTE(DataAddress);
464}
465
466/**
467 * Read a data byte from the controller, keyboard or mouse in polling mode.
468 */
469static NTSTATUS GetBytePoll(int DevType, PDEVEXT pDevExt, PUCHAR Byte)
470{
471 UCHAR byte;
472
473 ULONG i = 0;
474 UCHAR fMask = (DevType == MouDevType) ? (UCHAR)(OUTPUT_BUFFER_FULL | MOUSE_OUTPUT_BUFFER_FULL)
475 : (UCHAR) OUTPUT_BUFFER_FULL;
476 while ( (i < (ULONG)pDevExt->Cfg.PollingIterations)
477 && ((UCHAR)((byte = I8X_GET_STATUS_BYTE(pDevExt->DevRegs[i8042Cmd])) & fMask) != fMask))
478 {
479 if (byte & OUTPUT_BUFFER_FULL)
480 *Byte = I8X_GET_DATA_BYTE(pDevExt->DevRegs[i8042Dat]);
481 else
482 {
483 KeStallExecutionProcessor(pDevExt->Cfg.StallMicroseconds);
484 i++;
485 }
486 }
487 if (i >= (ULONG)pDevExt->Cfg.PollingIterations)
488 return STATUS_IO_TIMEOUT;
489
490 *Byte = I8X_GET_DATA_BYTE(pDevExt->DevRegs[i8042Dat]);
491 return STATUS_SUCCESS;
492}
493
494/**
495 * Send a command or data byte to the controller, keyboard or mouse.
496 */
497static NTSTATUS PutBytePoll(CCHAR PortType, BOOLEAN fWaitForAck, int AckDevType, PDEVEXT pDevExt, UCHAR Byte)
498{
499 NTSTATUS status;
500
501 if (AckDevType == MouDevType)
502 {
503 /* switch to AUX device */
504 PutBytePoll(i8042Cmd, FALSE /*=wait*/, NoDevice, pDevExt, I8042_WRITE_TO_AUXILIARY_DEVICE);
505 }
506
507 PUCHAR dataAddress = pDevExt->DevRegs[i8042Dat];
508 PUCHAR commandAddress = pDevExt->DevRegs[i8042Cmd];
509 for (unsigned j = 0; j < (unsigned)pDevExt->Cfg.iResend; j++)
510 {
511 unsigned i = 0;
512 while ( i++ < (ULONG)pDevExt->Cfg.PollingIterations
513 && (I8X_GET_STATUS_BYTE(commandAddress) & INPUT_BUFFER_FULL))
514 KeStallExecutionProcessor(pDevExt->Cfg.StallMicroseconds);
515 if (i >= (ULONG)pDevExt->Cfg.PollingIterations)
516 return STATUS_IO_TIMEOUT;
517
518 DrainOutBuf(dataAddress, commandAddress);
519
520 if (PortType == i8042Cmd)
521 I8X_PUT_COMMAND_BYTE(commandAddress, Byte);
522 else
523 I8X_PUT_DATA_BYTE(dataAddress, Byte);
524
525 if (!fWaitForAck)
526 return STATUS_SUCCESS;
527
528 BOOLEAN fKeepTrying = FALSE;
529 UCHAR byte;
530 while ((status = GetBytePoll(AckDevType, pDevExt, &byte)) == STATUS_SUCCESS)
531 {
532 if (byte == ACKNOWLEDGE)
533 break;
534 else if (byte == RESEND)
535 {
536 if (AckDevType == MouDevType)
537 PutBytePoll(i8042Cmd, FALSE /*=wait*/, NoDevice, pDevExt, I8042_WRITE_TO_AUXILIARY_DEVICE);
538 fKeepTrying = TRUE;
539 break;
540 }
541 }
542
543 if (!fKeepTrying)
544 return status;
545 }
546
547 return STATUS_IO_TIMEOUT;
548}
549
550/**
551 * Read a byte from controller, keyboard or mouse
552 */
553static VOID GetByteAsync(int DevType, PDEVEXT pDevExt, PUCHAR pByte)
554{
555 UCHAR byte;
556 UCHAR fMask;
557
558 ULONG i = 0;
559 fMask = (DevType == MouDevType)
560 ? (UCHAR) (OUTPUT_BUFFER_FULL | MOUSE_OUTPUT_BUFFER_FULL)
561 : (UCHAR) OUTPUT_BUFFER_FULL;
562
563 while ( i < (ULONG)pDevExt->Cfg.PollingIterations
564 && ((UCHAR)((byte = I8X_GET_STATUS_BYTE(pDevExt->DevRegs[i8042Cmd])) & fMask) != fMask))
565 {
566 if (byte & OUTPUT_BUFFER_FULL)
567 *pByte = I8X_GET_DATA_BYTE(pDevExt->DevRegs[i8042Dat]);
568 else
569 i++;
570 }
571 if (i >= (ULONG)pDevExt->Cfg.PollingIterations)
572 return;
573
574 *pByte = I8X_GET_DATA_BYTE(pDevExt->DevRegs[i8042Dat]);
575}
576
577/**
578 * Send a command or data byte to the controller, keyboard or mouse
579 * asynchronously.
580 */
581static VOID PutByteAsync(CCHAR PortType, PDEVEXT pDevExt, UCHAR Byte)
582{
583 unsigned i = 0;
584 while (I8X_GET_STATUS_BYTE(pDevExt->DevRegs[i8042Cmd]) & INPUT_BUFFER_FULL)
585 if (i++ >= (ULONG)pDevExt->Cfg.PollingIterations)
586 return;
587
588 if (PortType == i8042Cmd)
589 I8X_PUT_COMMAND_BYTE(pDevExt->DevRegs[i8042Cmd], Byte);
590 else
591 I8X_PUT_DATA_BYTE(pDevExt->DevRegs[i8042Dat], Byte);
592}
593
594/**
595 * Initiaze an I/O operation for the keyboard device.
596 */
597static VOID KbdStartIO(PVOID pCtx)
598{
599 PDEVICE_OBJECT pDevObj = (PDEVICE_OBJECT)pCtx;
600 PDEVEXT pDevExt = (PDEVEXT)pDevObj->DeviceExtension;
601
602 pDevExt->TimerCount = 3;
603
604 KBDSETPACKET keyboardPacket = pDevExt->KbdExt.CurrentOutput;
605
606 if (pDevExt->KbdExt.CurrentOutput.State == SendFirstByte)
607 PutByteAsync(i8042Dat, pDevExt, keyboardPacket.FirstByte);
608 else if (pDevExt->KbdExt.CurrentOutput.State == SendLastByte)
609 PutByteAsync(i8042Dat, pDevExt, keyboardPacket.LastByte);
610 else
611 ASSERT(FALSE);
612}
613
614static BOOLEAN KbdStartWrapper(PVOID pCtx)
615{
616 PDEVICE_OBJECT pDevObj = ((PKBDINITIATECTX)pCtx)->pDevObj;
617 PDEVEXT pDevExt = (PDEVEXT) pDevObj->DeviceExtension;
618 pDevExt->KbdExt.CurrentOutput.State = SendFirstByte;
619 pDevExt->KbdExt.CurrentOutput.FirstByte = ((PKBDINITIATECTX)pCtx)->FirstByte;
620 pDevExt->KbdExt.CurrentOutput.LastByte = ((PKBDINITIATECTX)pCtx)->LastByte;
621 pDevExt->KbdExt.ResendCount = 0;
622 KbdStartIO(pDevObj);
623 return TRUE;
624}
625
626static BOOLEAN DecTimer(PVOID pCtx)
627{
628 PTIMERCTX pTmCtx = (PTIMERCTX)pCtx;
629 PDEVICE_OBJECT pDevObj = pTmCtx->pDevObj;
630 PDEVEXT pDevExt = (PDEVEXT)pDevObj->DeviceExtension;
631
632 if (*(pTmCtx->TimerCounter) != -1)
633 (*(pTmCtx->TimerCounter))--;
634
635 pTmCtx->NewTimerCount = *(pTmCtx->TimerCounter);
636
637 if (*(pTmCtx->TimerCounter) == 0)
638 {
639 pDevExt->KbdExt.CurrentOutput.State = Idle;
640 pDevExt->KbdExt.ResendCount = 0;
641 }
642 return TRUE;
643}
644
645/**
646 * Perform an operation on the InterlockedDpcVariable.
647 */
648static BOOLEAN DpcVarOp(PVOID pCtx)
649{
650 PVAROPCTX pOpCtx = (PVAROPCTX)pCtx;
651 switch (pOpCtx->Operation)
652 {
653 case IncrementOperation:
654 (*pOpCtx->VariableAddress)++;
655 break;
656 case DecrementOperation:
657 (*pOpCtx->VariableAddress)--;
658 break;
659 case WriteOperation:
660 *pOpCtx->VariableAddress = *pOpCtx->NewValue;
661 break;
662 default:
663 ASSERT(FALSE);
664 break;
665 }
666
667 *(pOpCtx->NewValue) = *(pOpCtx->VariableAddress);
668 return TRUE;
669}
670
671static BOOLEAN GetDataQueuePtr(PVOID pCtx)
672{
673 PDEVEXT pDevExt = (PDEVEXT)((PGETDATAPTRCTX)pCtx)->pDevExt;
674 CCHAR DevType = (CCHAR)((PGETDATAPTRCTX)pCtx)->DevType;
675
676 if (DevType == KbdDevType)
677 {
678 ((PGETDATAPTRCTX)pCtx)->DataIn = pDevExt->KbdExt.DataIn;
679 ((PGETDATAPTRCTX)pCtx)->DataOut = pDevExt->KbdExt.DataOut;
680 ((PGETDATAPTRCTX)pCtx)->cInput = pDevExt->KbdExt.cInput;
681 }
682 else if (DevType == MouDevType)
683 {
684 ((PGETDATAPTRCTX)pCtx)->DataIn = pDevExt->MouExt.DataIn;
685 ((PGETDATAPTRCTX)pCtx)->DataOut = pDevExt->MouExt.DataOut;
686 ((PGETDATAPTRCTX)pCtx)->cInput = pDevExt->MouExt.cInput;
687 }
688 else
689 ASSERT(FALSE);
690 return TRUE;
691}
692
693static BOOLEAN InitDataQueue(PVOID pCtx)
694{
695 PDEVEXT pDevExt = (PDEVEXT)((PI8042INITDATACTX)pCtx)->pDevExt;
696 CCHAR DevType = (CCHAR) ((PI8042INITDATACTX)pCtx)->DevType;
697
698 if (DevType == KbdDevType)
699 {
700 pDevExt->KbdExt.cInput = 0;
701 pDevExt->KbdExt.DataIn = pDevExt->KbdExt.InputData;
702 pDevExt->KbdExt.DataOut = pDevExt->KbdExt.InputData;
703 }
704 else if (DevType == MouDevType)
705 {
706 pDevExt->MouExt.cInput = 0;
707 pDevExt->MouExt.DataIn = pDevExt->MouExt.InputData;
708 pDevExt->MouExt.DataOut = pDevExt->MouExt.InputData;
709 }
710 else
711 ASSERT(FALSE);
712 return TRUE;
713}
714
715static BOOLEAN SetDataQueuePtr(PVOID pCtx)
716{
717 PDEVEXT pDevExt = (PDEVEXT)((PSETDATAPTRCTX)pCtx)->pDevExt;
718 CCHAR DevType = (CCHAR) ((PSETDATAPTRCTX)pCtx)->DevType;
719
720 if (DevType == KbdDevType)
721 {
722 pDevExt->KbdExt.DataOut = (PKEYBOARD_INPUT_DATA)((PSETDATAPTRCTX)pCtx)->DataOut;
723 pDevExt->KbdExt.cInput -= ((PSETDATAPTRCTX)pCtx)->cInput;
724 }
725 else if (DevType == MouDevType)
726 {
727 pDevExt->MouExt.DataOut = (PMOUSE_INPUT_DATA)((PSETDATAPTRCTX)pCtx)->DataOut;
728 pDevExt->MouExt.cInput -= ((PSETDATAPTRCTX)pCtx)->cInput;
729 }
730 else
731 ASSERT(FALSE);
732 return TRUE;
733}
734
735/**
736 * DISPATCH_LEVEL IRQL: Complete requests.
737 */
738static VOID CompleteDpc(PKDPC Dpc, PDEVICE_OBJECT pDevObj, PIRP Irp, PVOID pCtx)
739{
740 NOREF(Dpc);
741 NOREF(pCtx);
742
743 PDEVEXT pDevExt = (PDEVEXT)pDevObj->DeviceExtension;
744
745 KeCancelTimer(&pDevExt->CommandTimer);
746
747 Irp = pDevObj->CurrentIrp;
748 ASSERT(Irp);
749
750 PIO_STACK_LOCATION irpSp = IoGetCurrentIrpStackLocation(Irp);
751 switch (irpSp->Parameters.DeviceIoControl.IoControlCode)
752 {
753 case IOCTL_KEYBOARD_SET_INDICATORS:
754 pDevExt->Cfg.KbdInd = *(PKEYBOARD_INDICATOR_PARAMETERS)Irp->AssociatedIrp.SystemBuffer;
755 break;
756
757 case IOCTL_KEYBOARD_SET_TYPEMATIC:
758 pDevExt->Cfg.KeyRepeatCurrent = *(PKEYBOARD_TYPEMATIC_PARAMETERS)Irp->AssociatedIrp.SystemBuffer;
759 break;
760
761 default:
762 break;
763 }
764
765 Irp->IoStatus.Status = STATUS_SUCCESS;
766 IoStartNextPacket(pDevObj, FALSE);
767 IoCompleteRequest (Irp, IO_KEYBOARD_INCREMENT);
768}
769
770static NTSTATUS I8042Flush(PDEVICE_OBJECT pDevObj, PIRP Irp)
771{
772 NOREF(pDevObj);
773 NOREF(Irp);
774
775 return STATUS_NOT_IMPLEMENTED;
776}
777
778/**
779 * Dispatch internal device control requests.
780 */
781static NTSTATUS I8042DevCtrl(PDEVICE_OBJECT pDevObj, PIRP Irp)
782{
783 NTSTATUS status;
784 I8042INITDATACTX initDataCtx;
785 PVOID pParams;
786 PKEYBOARD_ATTRIBUTES KbdAttr;
787 ULONG cbTrans;
788
789 PDEVEXT pDevExt = (PDEVEXT)pDevObj->DeviceExtension;
790
791 Irp->IoStatus.Information = 0;
792 PIO_STACK_LOCATION irpSp = IoGetCurrentIrpStackLocation(Irp);
793
794 switch (irpSp->Parameters.DeviceIoControl.IoControlCode)
795 {
796 case IOCTL_INTERNAL_KEYBOARD_CONNECT:
797 if ((pDevExt->HardwarePresent & KEYBOARD_HARDWARE_PRESENT) != KEYBOARD_HARDWARE_PRESENT)
798 {
799 status = STATUS_NO_SUCH_DEVICE;
800 break;
801 }
802 else if (pDevExt->KbdExt.ConnectData.ClassService)
803 {
804 status = STATUS_SHARING_VIOLATION;
805 break;
806 }
807 else if (irpSp->Parameters.DeviceIoControl.InputBufferLength < sizeof(CONNECT_DATA))
808 {
809 status = STATUS_INVALID_PARAMETER;
810 break;
811 }
812 pDevExt->KbdExt.ConnectData = *((PCONNECT_DATA) (irpSp->Parameters.DeviceIoControl.Type3InputBuffer));
813 initDataCtx.pDevExt = pDevExt;
814 initDataCtx.DevType = KbdDevType;
815 KeSynchronizeExecution(pDevExt->KbdIntObj, InitDataQueue, &initDataCtx);
816 status = STATUS_SUCCESS;
817 break;
818
819 case IOCTL_INTERNAL_MOUSE_CONNECT:
820 if ((pDevExt->HardwarePresent & MOUSE_HARDWARE_PRESENT) != MOUSE_HARDWARE_PRESENT)
821 {
822 status = STATUS_NO_SUCH_DEVICE;
823 break;
824 }
825 else if (pDevExt->MouExt.ConnectData.ClassService)
826 {
827 status = STATUS_SHARING_VIOLATION;
828 break;
829 }
830 else if (irpSp->Parameters.DeviceIoControl.InputBufferLength < sizeof(CONNECT_DATA))
831 {
832 status = STATUS_INVALID_PARAMETER;
833 break;
834 }
835 pDevExt->MouExt.ConnectData = *((PCONNECT_DATA) (irpSp->Parameters.DeviceIoControl.Type3InputBuffer));
836 initDataCtx.pDevExt = pDevExt;
837 initDataCtx.DevType = MouDevType;
838 KeSynchronizeExecution(pDevExt->MouIntObj, InitDataQueue, &initDataCtx);
839 status = STATUS_SUCCESS;
840 break;
841
842 case IOCTL_INTERNAL_KEYBOARD_DISCONNECT:
843 case IOCTL_INTERNAL_MOUSE_DISCONNECT:
844 status = STATUS_NOT_IMPLEMENTED;
845 break;
846
847 case IOCTL_INTERNAL_KEYBOARD_ENABLE:
848 case IOCTL_INTERNAL_KEYBOARD_DISABLE:
849 case IOCTL_INTERNAL_MOUSE_ENABLE:
850 case IOCTL_INTERNAL_MOUSE_DISABLE:
851 status = STATUS_PENDING;
852 break;
853
854 case IOCTL_KEYBOARD_QUERY_ATTRIBUTES:
855 if (irpSp->Parameters.DeviceIoControl.OutputBufferLength < sizeof(KEYBOARD_ATTRIBUTES))
856 status = STATUS_BUFFER_TOO_SMALL;
857 else
858 {
859 *(PKEYBOARD_ATTRIBUTES) Irp->AssociatedIrp.SystemBuffer = pDevExt->Cfg.KbdAttr;
860 Irp->IoStatus.Information = sizeof(KEYBOARD_ATTRIBUTES);
861 status = STATUS_SUCCESS;
862 }
863 break;
864
865 case IOCTL_KEYBOARD_QUERY_INDICATOR_TRANSLATION:
866 cbTrans = sizeof(KEYBOARD_INDICATOR_TRANSLATION)
867 + (sizeof(INDICATOR_LIST) * (pDevExt->Cfg.KbdAttr.NumberOfIndicators - 1));
868
869 if (irpSp->Parameters.DeviceIoControl.OutputBufferLength < cbTrans)
870 status = STATUS_BUFFER_TOO_SMALL;
871 else
872 {
873 ((PKEYBOARD_INDICATOR_TRANSLATION)
874 Irp->AssociatedIrp.SystemBuffer)->NumberOfIndicatorKeys = pDevExt->Cfg.KbdAttr.NumberOfIndicators;
875 RtlMoveMemory(((PKEYBOARD_INDICATOR_TRANSLATION)
876 Irp->AssociatedIrp.SystemBuffer)->IndicatorList, (PCHAR)s_aIndicators, cbTrans);
877
878 Irp->IoStatus.Information = cbTrans;
879 status = STATUS_SUCCESS;
880 }
881 break;
882
883 case IOCTL_KEYBOARD_QUERY_INDICATORS:
884 if (irpSp->Parameters.DeviceIoControl.OutputBufferLength < sizeof(KEYBOARD_INDICATOR_PARAMETERS))
885 status = STATUS_BUFFER_TOO_SMALL;
886 else
887 {
888 *(PKEYBOARD_INDICATOR_PARAMETERS)Irp->AssociatedIrp.SystemBuffer = pDevExt->Cfg.KbdInd;
889 Irp->IoStatus.Information = sizeof(KEYBOARD_INDICATOR_PARAMETERS);
890 status = STATUS_SUCCESS;
891 }
892 break;
893
894 case IOCTL_KEYBOARD_SET_INDICATORS:
895 if ( (irpSp->Parameters.DeviceIoControl.InputBufferLength < sizeof(KEYBOARD_INDICATOR_PARAMETERS))
896 || ( (((PKEYBOARD_INDICATOR_PARAMETERS)Irp->AssociatedIrp.SystemBuffer)->LedFlags
897 & ~(KEYBOARD_SCROLL_LOCK_ON | KEYBOARD_NUM_LOCK_ON | KEYBOARD_CAPS_LOCK_ON)) != 0))
898 status = STATUS_INVALID_PARAMETER;
899 else
900 status = STATUS_PENDING;
901 break;
902
903 case IOCTL_KEYBOARD_QUERY_TYPEMATIC:
904 if (irpSp->Parameters.DeviceIoControl.OutputBufferLength < sizeof(KEYBOARD_TYPEMATIC_PARAMETERS))
905 status = STATUS_BUFFER_TOO_SMALL;
906 else
907 {
908 *(PKEYBOARD_TYPEMATIC_PARAMETERS)Irp->AssociatedIrp.SystemBuffer = pDevExt->Cfg.KeyRepeatCurrent;
909 Irp->IoStatus.Information = sizeof(KEYBOARD_TYPEMATIC_PARAMETERS);
910 status = STATUS_SUCCESS;
911 }
912 break;
913
914 case IOCTL_KEYBOARD_SET_TYPEMATIC:
915 pParams = Irp->AssociatedIrp.SystemBuffer;
916 KbdAttr = &pDevExt->Cfg.KbdAttr;
917 if ( irpSp->Parameters.DeviceIoControl.InputBufferLength < sizeof(KEYBOARD_TYPEMATIC_PARAMETERS)
918 || ((PKEYBOARD_TYPEMATIC_PARAMETERS)pParams)->Rate < KbdAttr->KeyRepeatMinimum.Rate
919 || ((PKEYBOARD_TYPEMATIC_PARAMETERS)pParams)->Rate > KbdAttr->KeyRepeatMaximum.Rate
920 || ((PKEYBOARD_TYPEMATIC_PARAMETERS)pParams)->Delay < KbdAttr->KeyRepeatMinimum.Delay
921 || ((PKEYBOARD_TYPEMATIC_PARAMETERS)pParams)->Delay > KbdAttr->KeyRepeatMaximum.Delay)
922 status = STATUS_INVALID_PARAMETER;
923 else
924 status = STATUS_PENDING;
925 break;
926
927 case IOCTL_MOUSE_QUERY_ATTRIBUTES:
928 if (irpSp->Parameters.DeviceIoControl.OutputBufferLength < sizeof(MOUSE_ATTRIBUTES))
929 status = STATUS_BUFFER_TOO_SMALL;
930 else
931 {
932 *(PMOUSE_ATTRIBUTES) Irp->AssociatedIrp.SystemBuffer = pDevExt->Cfg.MouAttr;
933
934 Irp->IoStatus.Information = sizeof(MOUSE_ATTRIBUTES);
935 status = STATUS_SUCCESS;
936 }
937 break;
938
939 default:
940 status = STATUS_INVALID_DEVICE_REQUEST;
941 break;
942 }
943
944 Irp->IoStatus.Status = status;
945 if (status == STATUS_PENDING)
946 {
947 IoMarkIrpPending(Irp);
948 IoStartPacket(pDevObj, Irp, (PULONG)NULL, NULL);
949 }
950 else
951 IoCompleteRequest(Irp, IO_NO_INCREMENT);
952 return status;
953}
954
955/**
956 * Dispatch routine for create/open and close requests.
957 */
958static NTSTATUS I8042OpenClose(PDEVICE_OBJECT pDevObj, PIRP Irp)
959{
960 NOREF(pDevObj);
961
962 Irp->IoStatus.Status = STATUS_SUCCESS;
963 Irp->IoStatus.Information = 0;
964 IoCompleteRequest(Irp, IO_NO_INCREMENT);
965 return STATUS_SUCCESS;
966}
967
968/**
969 * DISPATCH_LEVEL IRQL: Complete requests that have exceeded the maximum
970 * number of retries.
971 */
972static VOID CtrlRetriesExceededDpc(PKDPC Dpc, PDEVICE_OBJECT pDevObj, PIRP Irp, PVOID pCtx)
973{
974 NOREF(Dpc);
975 NOREF(pCtx);
976 PDEVEXT pDevExt = (PDEVEXT)pDevObj->DeviceExtension;
977
978 Irp->IoStatus.Status = STATUS_IO_TIMEOUT;
979
980 IoStartNextPacket(pDevObj, FALSE);
981 IoCompleteRequest (Irp, IO_KEYBOARD_INCREMENT);
982}
983
984static UCHAR TypematicPeriod[] =
985{
986 31, 31, 28, 26, 23, 20, 18, 17, 15, 13, 12, 11, 10, 9,
987 9, 8, 7, 6, 5, 4, 4, 3, 3, 2, 2, 1, 1, 1
988};
989
990/**
991 * Convert typematic rate/delay to a value expected by the keyboard:
992 * - bit 7 is zero
993 * - bits 5...6 indicate the delay
994 * - bits 0...4 indicate the rate
995 */
996static UCHAR ConvertTypematic(USHORT Rate, USHORT Delay)
997{
998 UCHAR value = (UCHAR) ((Delay / 250) - 1);
999 value <<= 5;
1000 if (Rate <= 27)
1001 value |= TypematicPeriod[Rate];
1002
1003 return value;
1004}
1005
1006/**
1007 * Start an I/O operation for the device.
1008 */
1009static VOID I8042StartIo(PDEVICE_OBJECT pDevObj, PIRP Irp)
1010{
1011 KBDINITIATECTX keyboardInitiateContext;
1012 LARGE_INTEGER deltaTime;
1013 LONG interlockedResult;
1014
1015 PDEVEXT pDevExt = (PDEVEXT)pDevObj->DeviceExtension;
1016
1017 PIO_STACK_LOCATION irpSp = IoGetCurrentIrpStackLocation(Irp);
1018 switch (irpSp->Parameters.DeviceIoControl.IoControlCode)
1019 {
1020 case IOCTL_INTERNAL_KEYBOARD_ENABLE:
1021 interlockedResult = ASMAtomicIncU32(&pDevExt->KeyboardEnableCount);
1022 Irp->IoStatus.Status = STATUS_SUCCESS;
1023 IoStartNextPacket(pDevObj, FALSE);
1024 IoCompleteRequest(Irp, IO_KEYBOARD_INCREMENT);
1025 break;
1026
1027 case IOCTL_INTERNAL_KEYBOARD_DISABLE:
1028 if (pDevExt->KeyboardEnableCount == 0)
1029 Irp->IoStatus.Status = STATUS_DEVICE_DATA_ERROR;
1030 else
1031 {
1032 ASMAtomicDecU32(&pDevExt->KeyboardEnableCount);
1033 Irp->IoStatus.Status = STATUS_SUCCESS;
1034 }
1035 IoStartNextPacket(pDevObj, FALSE);
1036 IoCompleteRequest(Irp, IO_KEYBOARD_INCREMENT);
1037 break;
1038
1039 case IOCTL_INTERNAL_MOUSE_ENABLE:
1040 ASMAtomicIncU32(&pDevExt->MouseEnableCount);
1041 Irp->IoStatus.Status = STATUS_SUCCESS;
1042 IoStartNextPacket(pDevObj, FALSE);
1043 IoCompleteRequest(Irp, IO_MOUSE_INCREMENT);
1044 break;
1045
1046 case IOCTL_INTERNAL_MOUSE_DISABLE:
1047 if (pDevExt->MouseEnableCount == 0)
1048 Irp->IoStatus.Status = STATUS_DEVICE_DATA_ERROR;
1049 else
1050 {
1051 ASMAtomicDecU32(&pDevExt->MouseEnableCount);
1052 Irp->IoStatus.Status = STATUS_SUCCESS;
1053 }
1054 IoStartNextPacket(pDevObj, FALSE);
1055 IoCompleteRequest(Irp, IO_MOUSE_INCREMENT);
1056 break;
1057
1058 case IOCTL_KEYBOARD_SET_INDICATORS:
1059 keyboardInitiateContext.pDevObj = pDevObj;
1060 keyboardInitiateContext.FirstByte = SET_KEYBOARD_INDICATORS;
1061 keyboardInitiateContext.LastByte =
1062 (UCHAR) ((PKEYBOARD_INDICATOR_PARAMETERS)Irp->AssociatedIrp.SystemBuffer)->LedFlags;
1063 KeSynchronizeExecution(pDevExt->KbdIntObj, KbdStartWrapper, &keyboardInitiateContext);
1064 deltaTime.LowPart = (ULONG)(-10 * 1000 * 1000);
1065 deltaTime.HighPart = -1;
1066 KeSetTimer(&pDevExt->CommandTimer, deltaTime, &pDevExt->TimeOutDpc);
1067 break;
1068
1069 case IOCTL_KEYBOARD_SET_TYPEMATIC:
1070 keyboardInitiateContext.pDevObj = pDevObj;
1071 keyboardInitiateContext.FirstByte = SET_KEYBOARD_TYPEMATIC;
1072 keyboardInitiateContext.LastByte =
1073 ConvertTypematic(((PKEYBOARD_TYPEMATIC_PARAMETERS)Irp->AssociatedIrp.SystemBuffer)->Rate,
1074 ((PKEYBOARD_TYPEMATIC_PARAMETERS)Irp->AssociatedIrp.SystemBuffer)->Delay);
1075
1076 KeSynchronizeExecution(pDevExt->KbdIntObj, KbdStartWrapper, &keyboardInitiateContext);
1077 deltaTime.LowPart = (ULONG)(-10 * 1000 * 1000);
1078 deltaTime.HighPart = -1;
1079 KeSetTimer(&pDevExt->CommandTimer, deltaTime, &pDevExt->TimeOutDpc);
1080 break;
1081
1082 default:
1083 ASSERT(FALSE);
1084 break;
1085 }
1086}
1087
1088/**
1089 * Driver's command timeout routine.
1090 */
1091static VOID CtrlTimeoutDpc(PKDPC Dpc, PDEVICE_OBJECT pDevObj, PVOID SystemContext1, PVOID SystemContext2)
1092{
1093 PDEVEXT pDevExt = (PDEVEXT)pDevObj->DeviceExtension;
1094
1095 KIRQL cancelIrql;
1096 IoAcquireCancelSpinLock(&cancelIrql);
1097 PIRP irp = pDevObj->CurrentIrp;
1098 if (!irp)
1099 {
1100 IoReleaseCancelSpinLock(cancelIrql);
1101 return;
1102 }
1103 IoSetCancelRoutine(irp, NULL);
1104 IoReleaseCancelSpinLock(cancelIrql);
1105
1106 TIMERCTX timerContext;
1107 timerContext.pDevObj = pDevObj;
1108 timerContext.TimerCounter = &pDevExt->TimerCount;
1109 KeSynchronizeExecution(pDevExt->KbdIntObj, DecTimer, &timerContext);
1110
1111 if (timerContext.NewTimerCount == 0)
1112 {
1113 pDevObj->CurrentIrp->IoStatus.Information = 0;
1114 pDevObj->CurrentIrp->IoStatus.Status = STATUS_IO_TIMEOUT;
1115
1116 IoStartNextPacket(pDevObj, FALSE);
1117 IoCompleteRequest(irp, IO_KEYBOARD_INCREMENT);
1118 }
1119 else
1120 {
1121 LARGE_INTEGER deltaTime;
1122 deltaTime.LowPart = (ULONG)(-10 * 1000 * 1000);
1123 deltaTime.HighPart = -1;
1124 KeSetTimer(&pDevExt->CommandTimer, deltaTime, &pDevExt->TimeOutDpc);
1125 }
1126}
1127
1128/**
1129 * DISPATCH_LEVEL IRQL: Finish processing for keyboard interrupts.
1130 */
1131static VOID CtrlKbdIsrDpc(PKDPC Dpc, PDEVICE_OBJECT pDevObj, PIRP Irp, PVOID pCtx)
1132{
1133 NOREF(Dpc);
1134 NOREF(Irp);
1135 NOREF(pCtx);
1136
1137 PDEVEXT pDevExt = (PDEVEXT) pDevObj->DeviceExtension;
1138
1139 VAROPCTX opCtx;
1140 LONG interlockedResult;
1141 opCtx.VariableAddress = &pDevExt->DpcInterlockKeyboard;
1142 opCtx.Operation = IncrementOperation;
1143 opCtx.NewValue = &interlockedResult;
1144 KeSynchronizeExecution(pDevExt->KbdIntObj, DpcVarOp, (PVOID)&opCtx);
1145 BOOLEAN fContinue = (interlockedResult == 0) ? TRUE : FALSE;
1146
1147 while (fContinue)
1148 {
1149 ULONG cbNotConsumed = 0;
1150 ULONG inputDataConsumed = 0;
1151
1152 GETDATAPTRCTX getPtrCtx;
1153 getPtrCtx.pDevExt = pDevExt;
1154 getPtrCtx.DevType = KbdDevType;
1155 SETDATAPTRCTX setPtrCtx;
1156 setPtrCtx.pDevExt = pDevExt;
1157 setPtrCtx.DevType = KbdDevType;
1158 setPtrCtx.cInput = 0;
1159 KeSynchronizeExecution(pDevExt->KbdIntObj, GetDataQueuePtr, &getPtrCtx);
1160
1161 if (getPtrCtx.cInput)
1162 {
1163 PVOID classDeviceObject = pDevExt->KbdExt.ConnectData.ClassDeviceObject;
1164 PSERVICECALLBACK classService = pDevExt->KbdExt.ConnectData.ClassService;
1165 ASSERT(classService);
1166
1167 if (getPtrCtx.DataOut >= getPtrCtx.DataIn)
1168 {
1169 classService(classDeviceObject, getPtrCtx.DataOut, pDevExt->KbdExt.DataEnd, &inputDataConsumed);
1170 cbNotConsumed = (((PUCHAR) pDevExt->KbdExt.DataEnd - (PUCHAR) getPtrCtx.DataOut)
1171 / sizeof(KEYBOARD_INPUT_DATA)) - inputDataConsumed;
1172
1173 setPtrCtx.cInput += inputDataConsumed;
1174
1175 if (cbNotConsumed)
1176 {
1177 setPtrCtx.DataOut = ((PUCHAR)getPtrCtx.DataOut)
1178 + (inputDataConsumed * sizeof(KEYBOARD_INPUT_DATA));
1179 }
1180 else
1181 {
1182 setPtrCtx.DataOut = pDevExt->KbdExt.InputData;
1183 getPtrCtx.DataOut = setPtrCtx.DataOut;
1184 }
1185 }
1186
1187 if ( cbNotConsumed == 0
1188 && inputDataConsumed < getPtrCtx.cInput)
1189 {
1190 classService(classDeviceObject, getPtrCtx.DataOut, getPtrCtx.DataIn, &inputDataConsumed);
1191 cbNotConsumed = (((PUCHAR) getPtrCtx.DataIn - (PUCHAR) getPtrCtx.DataOut)
1192 / sizeof(KEYBOARD_INPUT_DATA)) - inputDataConsumed;
1193
1194 setPtrCtx.DataOut = ((PUCHAR)getPtrCtx.DataOut) +
1195 (inputDataConsumed * sizeof(KEYBOARD_INPUT_DATA));
1196 setPtrCtx.cInput += inputDataConsumed;
1197 }
1198
1199 KeSynchronizeExecution(pDevExt->KbdIntObj, SetDataQueuePtr, &setPtrCtx);
1200 }
1201
1202 if (cbNotConsumed)
1203 {
1204 opCtx.Operation = WriteOperation;
1205 interlockedResult = -1;
1206 opCtx.NewValue = &interlockedResult;
1207 KeSynchronizeExecution(pDevExt->KbdIntObj, DpcVarOp, &opCtx);
1208
1209 LARGE_INTEGER deltaTime;
1210 deltaTime.LowPart = (ULONG)(-10 * 1000 * 1000);
1211 deltaTime.HighPart = -1;
1212 KeSetTimer(&pDevExt->KbdExt.DataConsumptionTimer, deltaTime, &pDevExt->KeyboardIsrDpcRetry);
1213 fContinue = FALSE;
1214 }
1215 else
1216 {
1217 opCtx.Operation = DecrementOperation;
1218 opCtx.NewValue = &interlockedResult;
1219 KeSynchronizeExecution(pDevExt->KbdIntObj, DpcVarOp, &opCtx);
1220 if (interlockedResult != -1)
1221 {
1222 opCtx.Operation = WriteOperation;
1223 interlockedResult = 0;
1224 opCtx.NewValue = &interlockedResult;
1225 KeSynchronizeExecution(pDevExt->KbdIntObj, DpcVarOp, &opCtx);
1226 }
1227 else
1228 fContinue = FALSE;
1229 }
1230 }
1231}
1232
1233/**
1234 * DISPATCH_LEVEL IRQL: Finish processing of mouse interrupts
1235 */
1236static VOID CtrlMouIsrDpc(PKDPC Dpc, PDEVICE_OBJECT pDevObj, PIRP Irp, PVOID pCtx)
1237{
1238 NOREF(Dpc);
1239 NOREF(Irp);
1240 NOREF(pCtx);
1241
1242 PDEVEXT pDevExt = (PDEVEXT) pDevObj->DeviceExtension;
1243
1244 VAROPCTX opCtx;
1245 LONG interlockedResult;
1246 opCtx.VariableAddress = &pDevExt->DpcInterlockMouse;
1247 opCtx.Operation = IncrementOperation;
1248 opCtx.NewValue = &interlockedResult;
1249 KeSynchronizeExecution(pDevExt->MouIntObj, DpcVarOp, &opCtx);
1250 BOOLEAN fContinue = (interlockedResult == 0) ? TRUE : FALSE;
1251 while (fContinue)
1252 {
1253 ULONG cbNotConsumed = 0;
1254 ULONG inputDataConsumed = 0;
1255
1256 GETDATAPTRCTX getPtrCtx;
1257 getPtrCtx.pDevExt = pDevExt;
1258 getPtrCtx.DevType = MouDevType;
1259 SETDATAPTRCTX setPtrCtx;
1260 setPtrCtx.pDevExt = pDevExt;
1261 setPtrCtx.DevType = MouDevType;
1262 setPtrCtx.cInput = 0;
1263 KeSynchronizeExecution(pDevExt->MouIntObj, GetDataQueuePtr, &getPtrCtx);
1264 if (getPtrCtx.cInput)
1265 {
1266 PVOID classDeviceObject = pDevExt->MouExt.ConnectData.ClassDeviceObject;
1267 PSERVICECALLBACK classService = pDevExt->MouExt.ConnectData.ClassService;
1268 ASSERT(classService);
1269
1270 if (getPtrCtx.DataOut >= getPtrCtx.DataIn)
1271 {
1272 classService(classDeviceObject, getPtrCtx.DataOut, pDevExt->MouExt.DataEnd, &inputDataConsumed);
1273 cbNotConsumed = (((PUCHAR)pDevExt->MouExt.DataEnd - (PUCHAR) getPtrCtx.DataOut)
1274 / sizeof(MOUSE_INPUT_DATA)) - inputDataConsumed;
1275
1276 setPtrCtx.cInput += inputDataConsumed;
1277 if (cbNotConsumed)
1278 {
1279 setPtrCtx.DataOut = ((PUCHAR)getPtrCtx.DataOut) + (inputDataConsumed * sizeof(MOUSE_INPUT_DATA));
1280 }
1281 else
1282 {
1283 setPtrCtx.DataOut = pDevExt->MouExt.InputData;
1284 getPtrCtx.DataOut = setPtrCtx.DataOut;
1285 }
1286 }
1287
1288 if ( cbNotConsumed == 0
1289 && inputDataConsumed < getPtrCtx.cInput)
1290 {
1291 classService(classDeviceObject, getPtrCtx.DataOut, getPtrCtx.DataIn, &inputDataConsumed);
1292 cbNotConsumed = (((PUCHAR) getPtrCtx.DataIn - (PUCHAR) getPtrCtx.DataOut)
1293 / sizeof(MOUSE_INPUT_DATA)) - inputDataConsumed;
1294
1295 setPtrCtx.DataOut = ((PUCHAR)getPtrCtx.DataOut) + (inputDataConsumed * sizeof(MOUSE_INPUT_DATA));
1296 setPtrCtx.cInput += inputDataConsumed;
1297 }
1298 KeSynchronizeExecution(pDevExt->MouIntObj, SetDataQueuePtr, &setPtrCtx);
1299 }
1300
1301 if (cbNotConsumed)
1302 {
1303 opCtx.Operation = WriteOperation;
1304 interlockedResult = -1;
1305 opCtx.NewValue = &interlockedResult;
1306 KeSynchronizeExecution(pDevExt->MouIntObj, DpcVarOp, &opCtx);
1307
1308 LARGE_INTEGER deltaTime;
1309 deltaTime.LowPart = (ULONG)(-10 * 1000 * 1000);
1310 deltaTime.HighPart = -1;
1311 KeSetTimer(&pDevExt->MouExt.DataConsumptionTimer, deltaTime, &pDevExt->MouseIsrDpcRetry);
1312 fContinue = FALSE;
1313 }
1314 else
1315 {
1316 opCtx.Operation = DecrementOperation;
1317 opCtx.NewValue = &interlockedResult;
1318 KeSynchronizeExecution(pDevExt->MouIntObj, DpcVarOp, &opCtx);
1319
1320 if (interlockedResult != -1)
1321 {
1322 opCtx.Operation = WriteOperation;
1323 interlockedResult = 0;
1324 opCtx.NewValue = &interlockedResult;
1325 KeSynchronizeExecution(pDevExt->MouIntObj, DpcVarOp, &opCtx);
1326 }
1327 else
1328 fContinue = FALSE;
1329 }
1330 }
1331}
1332
1333/**
1334 * Interrupt service routine for the mouse device.
1335 */
1336static BOOLEAN MouIntHandler(PKINTERRUPT Interrupt, PVOID pCtx)
1337{
1338 UCHAR uPrevSignAndOverflow;
1339
1340 NOREF(Interrupt);
1341
1342 PDEVICE_OBJECT pDevObj = (PDEVICE_OBJECT)pCtx;
1343 PDEVEXT pDevExt = (PDEVEXT)pDevObj->DeviceExtension;
1344
1345 if ((I8X_GET_STATUS_BYTE(pDevExt->DevRegs[i8042Cmd]) & (OUTPUT_BUFFER_FULL|MOUSE_OUTPUT_BUFFER_FULL))
1346 != (OUTPUT_BUFFER_FULL|MOUSE_OUTPUT_BUFFER_FULL))
1347 {
1348 KeStallExecutionProcessor(10);
1349 if ((I8X_GET_STATUS_BYTE(pDevExt->DevRegs[i8042Cmd]) & (OUTPUT_BUFFER_FULL|MOUSE_OUTPUT_BUFFER_FULL))
1350 != (OUTPUT_BUFFER_FULL|MOUSE_OUTPUT_BUFFER_FULL))
1351 return FALSE;
1352 }
1353
1354 UCHAR byte;
1355 GetByteAsync(MouDevType, pDevExt, &byte);
1356
1357 if ( pDevExt->MouExt.LastByteReceived == 0xaa
1358 && byte == 0x00)
1359 {
1360 pDevExt->HardwarePresent &= ~WHEELMOUSE_HARDWARE_PRESENT;
1361 pDevExt->Cfg.MouAttr.NumberOfButtons = 2;
1362
1363 PutByteAsync(i8042Cmd, pDevExt, I8042_WRITE_TO_AUXILIARY_DEVICE);
1364 PutByteAsync(i8042Dat, pDevExt, ENABLE_MOUSE_TRANSMISSION);
1365
1366 pDevExt->MouExt.InputState = MouseExpectingACK;
1367 }
1368
1369 pDevExt->MouExt.LastByteReceived = byte;
1370
1371 LARGE_INTEGER tickDelta, newTick;
1372 KeQueryTickCount(&newTick);
1373 tickDelta.QuadPart = newTick.QuadPart - pDevExt->MouExt.PreviousTick.QuadPart;
1374 if ( pDevExt->MouExt.InputState != MouseIdle
1375 && pDevExt->MouExt.InputState != MouseExpectingACK
1376 && (tickDelta.LowPart >= pDevExt->MouExt.SynchTickCount || tickDelta.HighPart != 0))
1377 pDevExt->MouExt.InputState = MouseIdle;
1378 pDevExt->MouExt.PreviousTick = newTick;
1379
1380 switch (pDevExt->MouExt.InputState)
1381 {
1382 case MouseIdle:
1383 {
1384 UCHAR fPrevBtns = pDevExt->MouExt.PreviousButtons;
1385 pDevExt->MouExt.CurrentInput.ButtonFlags = 0;
1386 pDevExt->MouExt.CurrentInput.ButtonData = 0;
1387
1388 if ((!(fPrevBtns & LEFT_BUTTON_DOWN)) && (byte & LEFT_BUTTON_DOWN))
1389 pDevExt->MouExt.CurrentInput.ButtonFlags |= MOUSE_LEFT_BUTTON_DOWN;
1390 else if ((fPrevBtns & LEFT_BUTTON_DOWN) && !(byte & LEFT_BUTTON_DOWN))
1391 pDevExt->MouExt.CurrentInput.ButtonFlags |= MOUSE_LEFT_BUTTON_UP;
1392 if ((!(fPrevBtns & RIGHT_BUTTON_DOWN)) && (byte & RIGHT_BUTTON_DOWN))
1393 pDevExt->MouExt.CurrentInput.ButtonFlags |= MOUSE_RIGHT_BUTTON_DOWN;
1394 else if ((fPrevBtns & RIGHT_BUTTON_DOWN) && !(byte & RIGHT_BUTTON_DOWN))
1395 pDevExt->MouExt.CurrentInput.ButtonFlags |= MOUSE_RIGHT_BUTTON_UP;
1396 if ((!(fPrevBtns & MIDDLE_BUTTON_DOWN)) && (byte & MIDDLE_BUTTON_DOWN))
1397 pDevExt->MouExt.CurrentInput.ButtonFlags |= MOUSE_MIDDLE_BUTTON_DOWN;
1398 else if ((fPrevBtns & MIDDLE_BUTTON_DOWN) && !(byte & MIDDLE_BUTTON_DOWN))
1399 pDevExt->MouExt.CurrentInput.ButtonFlags |= MOUSE_MIDDLE_BUTTON_UP;
1400
1401 pDevExt->MouExt.PreviousButtons = byte & (RIGHT_BUTTON_DOWN|MIDDLE_BUTTON_DOWN|LEFT_BUTTON_DOWN);
1402 pDevExt->MouExt.uCurrSignAndOverflow = (UCHAR) (byte & MOUSE_SIGN_OVERFLOW_MASK);
1403 pDevExt->MouExt.InputState = XMovement;
1404 break;
1405 }
1406
1407 case XMovement:
1408 {
1409 if (pDevExt->MouExt.uCurrSignAndOverflow & X_OVERFLOW)
1410 {
1411 uPrevSignAndOverflow = pDevExt->MouExt.uPrevSignAndOverflow;
1412 if (uPrevSignAndOverflow & X_OVERFLOW)
1413 {
1414 if ((uPrevSignAndOverflow & X_DATA_SIGN) != (pDevExt->MouExt.uCurrSignAndOverflow & X_DATA_SIGN))
1415 pDevExt->MouExt.uCurrSignAndOverflow ^= X_DATA_SIGN;
1416 }
1417 if (pDevExt->MouExt.uCurrSignAndOverflow & X_DATA_SIGN)
1418 pDevExt->MouExt.CurrentInput.LastX = MOUSE_MAXIMUM_NEGATIVE_DELTA;
1419 else
1420 pDevExt->MouExt.CurrentInput.LastX = MOUSE_MAXIMUM_POSITIVE_DELTA;
1421 }
1422 else
1423 {
1424 pDevExt->MouExt.CurrentInput.LastX = (ULONG) byte;
1425 if (pDevExt->MouExt.uCurrSignAndOverflow & X_DATA_SIGN)
1426 pDevExt->MouExt.CurrentInput.LastX |= MOUSE_MAXIMUM_NEGATIVE_DELTA;
1427 }
1428 pDevExt->MouExt.InputState = YMovement;
1429 break;
1430 }
1431
1432 case YMovement:
1433 {
1434 if (pDevExt->MouExt.uCurrSignAndOverflow & Y_OVERFLOW)
1435 {
1436 uPrevSignAndOverflow = pDevExt->MouExt.uPrevSignAndOverflow;
1437 if (uPrevSignAndOverflow & Y_OVERFLOW)
1438 {
1439 if ((uPrevSignAndOverflow & Y_DATA_SIGN) != (pDevExt->MouExt.uCurrSignAndOverflow & Y_DATA_SIGN))
1440 pDevExt->MouExt.uCurrSignAndOverflow ^= Y_DATA_SIGN;
1441 }
1442 if (pDevExt->MouExt.uCurrSignAndOverflow & Y_DATA_SIGN)
1443 pDevExt->MouExt.CurrentInput.LastY = MOUSE_MAXIMUM_POSITIVE_DELTA;
1444 else
1445 pDevExt->MouExt.CurrentInput.LastY = MOUSE_MAXIMUM_NEGATIVE_DELTA;
1446 }
1447 else
1448 {
1449 pDevExt->MouExt.CurrentInput.LastY = (ULONG) byte;
1450 if (pDevExt->MouExt.uCurrSignAndOverflow & Y_DATA_SIGN)
1451 pDevExt->MouExt.CurrentInput.LastY |= MOUSE_MAXIMUM_NEGATIVE_DELTA;
1452 pDevExt->MouExt.CurrentInput.LastY = -pDevExt->MouExt.CurrentInput.LastY;
1453 }
1454 pDevExt->MouExt.uPrevSignAndOverflow = pDevExt->MouExt.uCurrSignAndOverflow;
1455
1456 if (pDevExt->HardwarePresent & WHEELMOUSE_HARDWARE_PRESENT)
1457 pDevExt->MouExt.InputState = ZMovement;
1458 else
1459 {
1460 pDevExt->MouExt.CurrentInput.Flags = MOUSE_MOVE_RELATIVE;
1461
1462 {
1463 VMMDevReqMouseStatus *pReq = pDevExt->pReq;
1464 if (pReq)
1465 {
1466 int rc = VbglGRPerform (&pReq->header);
1467 if (RT_SUCCESS(rc))
1468 {
1469 if (pReq->mouseFeatures & VMMDEV_MOUSE_HOST_WANTS_ABSOLUTE)
1470 {
1471 /* make it an absolute move */
1472 pDevExt->MouExt.CurrentInput.Flags = MOUSE_MOVE_ABSOLUTE;
1473 pDevExt->MouExt.CurrentInput.LastX = pReq->pointerXPos;
1474 pDevExt->MouExt.CurrentInput.LastY = pReq->pointerYPos;
1475 }
1476 }
1477 else
1478 Log(("VBoxMouseNT: ERROR querying mouse capabilities from VMMDev. rc = %Rrc\n", rc));
1479 }
1480 }
1481 QueueInput(pDevObj);
1482 pDevExt->MouExt.InputState = MouseIdle;
1483 }
1484 break;
1485 }
1486
1487 case ZMovement:
1488 {
1489#if 0
1490 if (byte && pDevExt->MouExt.CurrentInput.Buttons == 0)
1491#else
1492 if (byte)
1493#endif
1494 {
1495 if (byte & 0x80)
1496 pDevExt->MouExt.CurrentInput.ButtonData = 0x0078;
1497 else
1498 pDevExt->MouExt.CurrentInput.ButtonData = 0xFF88;
1499 pDevExt->MouExt.CurrentInput.ButtonFlags |= MOUSE_WHEEL;
1500 }
1501
1502 pDevExt->MouExt.CurrentInput.Flags = MOUSE_MOVE_RELATIVE;
1503
1504 {
1505 VMMDevReqMouseStatus *pReq = pDevExt->pReq;
1506 if (pReq)
1507 {
1508 int rc = VbglGRPerform(&pReq->header);
1509 if (RT_SUCCESS(rc))
1510 {
1511 if (pReq->mouseFeatures & VMMDEV_MOUSE_HOST_WANTS_ABSOLUTE)
1512 {
1513 /* make it an absolute move */
1514 pDevExt->MouExt.CurrentInput.Flags = MOUSE_MOVE_ABSOLUTE;
1515 pDevExt->MouExt.CurrentInput.LastX = pReq->pointerXPos;
1516 pDevExt->MouExt.CurrentInput.LastY = pReq->pointerYPos;
1517 }
1518 }
1519 else
1520 Log(("VBoxMouseNT: ERROR querying mouse capabilities from VMMDev. rc = %Rrc\n", rc));
1521 }
1522 }
1523
1524 QueueInput(pDevObj);
1525 pDevExt->MouExt.InputState = MouseIdle;
1526 break;
1527 }
1528
1529 case MouseExpectingACK:
1530 {
1531 if (byte == ACKNOWLEDGE)
1532 pDevExt->MouExt.InputState = MouseIdle;
1533 else if (byte == RESEND)
1534 {
1535 PutByteAsync(i8042Cmd, pDevExt, I8042_WRITE_TO_AUXILIARY_DEVICE);
1536 PutByteAsync(i8042Dat, pDevExt, ENABLE_MOUSE_TRANSMISSION);
1537 }
1538 break;
1539 }
1540
1541 default:
1542 {
1543 ASSERT(FALSE);
1544 break;
1545 }
1546 }
1547
1548 return TRUE;
1549}
1550
1551/**
1552 * Interrupt service routine.
1553 */
1554static BOOLEAN KbdIntHandler(PKINTERRUPT Interrupt, PVOID pCtx)
1555{
1556 NOREF(Interrupt);
1557
1558 PDEVICE_OBJECT pDevObj = (PDEVICE_OBJECT)pCtx;
1559 PDEVEXT pDevExt = (PDEVEXT) pDevObj->DeviceExtension;
1560
1561 if ((I8X_GET_STATUS_BYTE(pDevExt->DevRegs[i8042Cmd]) & (OUTPUT_BUFFER_FULL|MOUSE_OUTPUT_BUFFER_FULL))
1562 != OUTPUT_BUFFER_FULL)
1563 {
1564 for (unsigned i = 0; i < (ULONG)pDevExt->Cfg.PollStatusIterations; i++)
1565 {
1566 KeStallExecutionProcessor(1);
1567 if ((I8X_GET_STATUS_BYTE(pDevExt->DevRegs[i8042Cmd]) & (OUTPUT_BUFFER_FULL|MOUSE_OUTPUT_BUFFER_FULL))
1568 == (OUTPUT_BUFFER_FULL))
1569 break;
1570 }
1571
1572 if ((I8X_GET_STATUS_BYTE(pDevExt->DevRegs[i8042Cmd]) & (OUTPUT_BUFFER_FULL|MOUSE_OUTPUT_BUFFER_FULL))
1573 != (OUTPUT_BUFFER_FULL))
1574 {
1575 if (pDevExt->KeyboardEnableCount == 0)
1576 UCHAR scanCode = I8X_GET_DATA_BYTE(pDevExt->DevRegs[i8042Dat]);
1577 return FALSE;
1578 }
1579 }
1580
1581 UCHAR scanCode;
1582 GetByteAsync(KbdDevType, pDevExt, &scanCode);
1583 switch (scanCode)
1584 {
1585 case RESEND:
1586 if (pDevExt->TimerCount == 0)
1587 break;
1588 pDevExt->TimerCount = -1;
1589
1590 if ( pDevExt->KbdExt.CurrentOutput.State==Idle
1591 || !pDevObj->CurrentIrp)
1592 goto ScanCodeCase;
1593 else if (pDevExt->KbdExt.ResendCount < pDevExt->Cfg.iResend)
1594 {
1595 pDevExt->KbdExt.ResendCount++;
1596 KbdStartIO(pDevObj);
1597 }
1598 else
1599 {
1600 pDevExt->KbdExt.CurrentOutput.State = Idle;
1601 KeInsertQueueDpc(&pDevExt->RetriesExceededDpc, pDevObj->CurrentIrp, NULL);
1602 }
1603 break;
1604
1605 case ACKNOWLEDGE:
1606 if (pDevExt->TimerCount == 0)
1607 break;
1608
1609 pDevExt->TimerCount = -1;
1610
1611 pDevExt->KbdExt.ResendCount = 0;
1612 if (pDevExt->KbdExt.CurrentOutput.State == SendFirstByte)
1613 {
1614 pDevExt->KbdExt.CurrentOutput.State = SendLastByte;
1615 KbdStartIO(pDevObj);
1616 }
1617 else if (pDevExt->KbdExt.CurrentOutput.State == SendLastByte)
1618 {
1619 pDevExt->KbdExt.CurrentOutput.State = Idle;
1620 IoRequestDpc(pDevObj, pDevObj->CurrentIrp, NULL);
1621 }
1622 break;
1623
1624 ScanCodeCase:
1625 default:
1626 {
1627 PKEYBOARD_INPUT_DATA input = &pDevExt->KbdExt.CurrentInput;
1628 KBDSCANSTATE *pScanState = &pDevExt->KbdExt.CurrentScanState;
1629
1630 if (scanCode == (UCHAR) 0xFF)
1631 {
1632 input->MakeCode = KEYBOARD_OVERRUN_MAKE_CODE;
1633 input->Flags = 0;
1634 *pScanState = Normal;
1635 }
1636 else
1637 {
1638 switch (*pScanState)
1639 {
1640 case Normal:
1641 if (scanCode == (UCHAR) 0xE0)
1642 {
1643 input->Flags |= KEY_E0;
1644 *pScanState = GotE0;
1645 break;
1646 }
1647 else if (scanCode == (UCHAR) 0xE1)
1648 {
1649 input->Flags |= KEY_E1;
1650 *pScanState = GotE1;
1651 break;
1652 }
1653 /* fall through */
1654 case GotE0:
1655 case GotE1:
1656 if (scanCode > 0x7F)
1657 {
1658 input->MakeCode = scanCode & 0x7F;
1659 input->Flags |= KEY_BREAK;
1660 }
1661 else
1662 input->MakeCode = scanCode;
1663 *pScanState = Normal;
1664 break;
1665
1666 default:
1667 ASSERT(FALSE);
1668 break;
1669 }
1670 }
1671
1672 if (*pScanState == Normal)
1673 {
1674 if (pDevExt->KeyboardEnableCount)
1675 {
1676 pDevExt->KbdExt.CurrentInput.UnitId = pDevExt->KbdExt.UnitId;
1677 if (!KbdDataToQueue(&pDevExt->KbdExt, input))
1678 {
1679 }
1680 else if (pDevExt->DpcInterlockKeyboard >= 0)
1681 pDevExt->DpcInterlockKeyboard++;
1682 else
1683 KeInsertQueueDpc(&pDevExt->KeyboardIsrDpc, pDevObj->CurrentIrp, NULL);
1684 }
1685 input->Flags = 0;
1686 }
1687 break;
1688 }
1689 }
1690 return TRUE;
1691}
1692
1693static NTSTATUS MouEnableTrans(PDEVICE_OBJECT pDevObj)
1694{
1695 NTSTATUS status;
1696
1697 PDEVEXT pDevExt = (PDEVEXT) pDevObj->DeviceExtension;
1698 status = PutBytePoll(i8042Dat, FALSE /*=wait*/, MouDevType, pDevExt, ENABLE_MOUSE_TRANSMISSION);
1699 return status;
1700}
1701
1702/**
1703 * Configuration information for the keyboard.
1704 */
1705static VOID KbdGetRegstry(PINITEXT pInit, PUNICODE_STRING RegistryPath,
1706 PUNICODE_STRING KeyboardDeviceName, PUNICODE_STRING PointerDeviceName)
1707{
1708 PDEVEXT pDevExt = &pInit->DevExt;
1709 for (unsigned i = 0; i < MaximumInterfaceType; i++)
1710 {
1711 INTERFACE_TYPE interfaceType = (INTERFACE_TYPE)i;
1712 CONFIGURATION_TYPE controllerType = KeyboardController;
1713 CONFIGURATION_TYPE peripheralType = KeyboardPeripheral;
1714 NTSTATUS status = IoQueryDeviceDescription(&interfaceType, NULL,
1715 &controllerType, NULL,
1716 &peripheralType, NULL,
1717 KbdCallOut, pInit);
1718
1719 if (pDevExt->HardwarePresent & KEYBOARD_HARDWARE_PRESENT)
1720 {
1721 HwGetRegstry(pInit, RegistryPath, KeyboardDeviceName, PointerDeviceName);
1722
1723 PI8042CFGINF pCfg = &pInit->DevExt.Cfg;
1724 PKEYBOARD_ID keyboardId = &pCfg->KbdAttr.KeyboardIdentifier;
1725 if (!ENHANCED_KEYBOARD(*keyboardId))
1726 pCfg->PollingIterations = pCfg->PollingIterationsMaximum;
1727
1728 pCfg->KbdAttr.NumberOfFunctionKeys = s_aKeybType[keyboardId->Type-1].cFunctionKeys;
1729 pCfg->KbdAttr.NumberOfIndicators = s_aKeybType[keyboardId->Type-1].cIndicators;
1730 pCfg->KbdAttr.NumberOfKeysTotal = s_aKeybType[keyboardId->Type-1].cKeysTotal;
1731 pCfg->KbdAttr.KeyboardMode = 1;
1732 pCfg->KbdAttr.KeyRepeatMinimum.Rate = 2;
1733 pCfg->KbdAttr.KeyRepeatMinimum.Delay = 250;
1734 pCfg->KbdAttr.KeyRepeatMaximum.Rate = 30;
1735 pCfg->KbdAttr.KeyRepeatMaximum.Delay = 1000;
1736 pCfg->KeyRepeatCurrent.Rate = 30;
1737 pCfg->KeyRepeatCurrent.Delay = 250;
1738 break;
1739 }
1740 }
1741}
1742
1743/**
1744 * Retrieve the configuration information for the mouse.
1745 */
1746static VOID MouGetRegstry(PINITEXT pInit, PUNICODE_STRING RegistryPath,
1747 PUNICODE_STRING KeyboardDeviceName, PUNICODE_STRING PointerDeviceName)
1748{
1749 PDEVEXT pDevExt = &pInit->DevExt;
1750 NTSTATUS status = STATUS_SUCCESS;
1751 INTERFACE_TYPE interfaceType;
1752 CONFIGURATION_TYPE controllerType = PointerController;
1753 CONFIGURATION_TYPE peripheralType = PointerPeripheral;
1754
1755 for (unsigned i = 0; i < MaximumInterfaceType; i++)
1756 {
1757 interfaceType = (INTERFACE_TYPE)i;
1758 status = IoQueryDeviceDescription(&interfaceType, NULL,
1759 &controllerType, NULL,
1760 &peripheralType, NULL,
1761 MouCallOut, pInit);
1762
1763 if (pInit->DevExt.HardwarePresent & MOUSE_HARDWARE_PRESENT)
1764 {
1765 if (!(pInit->DevExt.HardwarePresent & KEYBOARD_HARDWARE_PRESENT))
1766 HwGetRegstry(pInit, RegistryPath, KeyboardDeviceName, PointerDeviceName);
1767 pInit->DevExt.Cfg.MouAttr.MouseIdentifier = MOUSE_I8042_HARDWARE;
1768 break;
1769 }
1770 }
1771}
1772
1773/**
1774 * Initialize the driver.
1775 */
1776NTSTATUS DriverEntry(PDRIVER_OBJECT pDrvObj, PUNICODE_STRING RegistryPath)
1777{
1778 PDEVICE_OBJECT pPortDevObj = NULL;
1779 NTSTATUS status = STATUS_SUCCESS;
1780 KIRQL IrqlCoord = 0;
1781 ULONG IntVecKbd;
1782 ULONG IntVecMou;
1783 KIRQL IrqlKbd;
1784 KIRQL IrqlMou;
1785 KAFFINITY AffKbd;
1786 KAFFINITY AffMou;
1787 ULONG addressSpace;
1788 PHYSICAL_ADDRESS Phys;
1789 BOOLEAN fConflict;
1790
1791 UNICODE_STRING KbdNameFull;
1792 UNICODE_STRING MouNameFull;
1793 UNICODE_STRING KbdNameBase;
1794 UNICODE_STRING MouNameBase;
1795 UNICODE_STRING DevNameSuff;
1796 UNICODE_STRING resourceDeviceClass;
1797 UNICODE_STRING registryPath;
1798
1799#define NAME_MAX 256
1800 WCHAR keyboardBuffer[NAME_MAX];
1801 WCHAR pointerBuffer[NAME_MAX];
1802
1803 LogFlow(("VBoxMouseNT::DriverEntry: enter\n"));
1804
1805 PINITEXT pInit = (PINITEXT)ExAllocatePool(NonPagedPool, sizeof(INITEXT));
1806 if (!pInit)
1807 {
1808 status = STATUS_UNSUCCESSFUL;
1809 goto fail;
1810 }
1811
1812 RtlZeroMemory(pInit, sizeof(INITEXT));
1813 KbdNameFull.MaximumLength = 0;
1814 KbdNameFull.Length = 0;
1815 MouNameFull.MaximumLength = 0;
1816 MouNameFull.Length = 0;
1817 DevNameSuff.MaximumLength = 0;
1818 DevNameSuff.Length = 0;
1819 resourceDeviceClass.MaximumLength = 0;
1820 resourceDeviceClass.Length = 0;
1821 registryPath.MaximumLength = 0;
1822 RtlZeroMemory(keyboardBuffer, NAME_MAX * sizeof(WCHAR));
1823 KbdNameBase.Buffer = keyboardBuffer;
1824 KbdNameBase.Length = 0;
1825 KbdNameBase.MaximumLength = NAME_MAX * sizeof(WCHAR);
1826 RtlZeroMemory(pointerBuffer, NAME_MAX * sizeof(WCHAR));
1827 MouNameBase.Buffer = pointerBuffer;
1828 MouNameBase.Length = 0;
1829 MouNameBase.MaximumLength = NAME_MAX * sizeof(WCHAR);
1830
1831 registryPath.Buffer = (PWSTR)ExAllocatePool(PagedPool, RegistryPath->Length + sizeof(UNICODE_NULL));
1832 if (!registryPath.Buffer)
1833 {
1834 status = STATUS_UNSUCCESSFUL;
1835 goto fail;
1836 }
1837 else
1838 {
1839 registryPath.Length = RegistryPath->Length + sizeof(UNICODE_NULL);
1840 registryPath.MaximumLength = registryPath.Length;
1841
1842 RtlZeroMemory(registryPath.Buffer, registryPath.Length);
1843 RtlMoveMemory(registryPath.Buffer, RegistryPath->Buffer, RegistryPath->Length);
1844 }
1845
1846 KbdGetRegstry(pInit, &registryPath, &KbdNameBase, &MouNameBase);
1847 MouGetRegstry(pInit, &registryPath, &KbdNameBase, &MouNameBase);
1848 if (pInit->DevExt.HardwarePresent == 0)
1849 {
1850 status = STATUS_NO_SUCH_DEVICE;
1851 goto fail;
1852 }
1853 else if (!(pInit->DevExt.HardwarePresent & KEYBOARD_HARDWARE_PRESENT))
1854 status = STATUS_NO_SUCH_DEVICE;
1855
1856 RtlInitUnicodeString(&DevNameSuff, NULL);
1857
1858 DevNameSuff.MaximumLength = (KEYBOARD_PORTS_MAXIMUM > POINTER_PORTS_MAXIMUM)
1859 ? KEYBOARD_PORTS_MAXIMUM * sizeof(WCHAR)
1860 : POINTER_PORTS_MAXIMUM * sizeof(WCHAR);
1861 DevNameSuff.MaximumLength += sizeof(UNICODE_NULL);
1862 DevNameSuff.Buffer = (PWSTR)ExAllocatePool(PagedPool, DevNameSuff.MaximumLength);
1863 if (!DevNameSuff.Buffer)
1864 {
1865 status = STATUS_UNSUCCESSFUL;
1866 goto fail;
1867 }
1868
1869 RtlZeroMemory(DevNameSuff.Buffer, DevNameSuff.MaximumLength);
1870
1871 RtlInitUnicodeString(&KbdNameFull, NULL);
1872 KbdNameFull.MaximumLength = sizeof(L"\\Device\\") + KbdNameBase.Length + DevNameSuff.MaximumLength;
1873 KbdNameFull.Buffer = (PWSTR)ExAllocatePool(PagedPool, KbdNameFull.MaximumLength);
1874 if (!KbdNameFull.Buffer)
1875 {
1876 status = STATUS_UNSUCCESSFUL;
1877 goto fail;
1878 }
1879
1880 RtlZeroMemory(KbdNameFull.Buffer, KbdNameFull.MaximumLength);
1881 RtlAppendUnicodeToString(&KbdNameFull, L"\\Device\\");
1882 RtlAppendUnicodeToString(&KbdNameFull, KbdNameBase.Buffer);
1883
1884 for (unsigned i = 0; i < KEYBOARD_PORTS_MAXIMUM; i++)
1885 {
1886 status = RtlIntegerToUnicodeString(i, 10, &DevNameSuff);
1887 if (!NT_SUCCESS(status))
1888 break;
1889 RtlAppendUnicodeStringToString(&KbdNameFull, &DevNameSuff);
1890
1891 LogFlow(("VBoxMouseNT::DriverEntry: Creating device object named %S\n", KbdNameFull.Buffer));
1892
1893 status = IoCreateDevice(pDrvObj, sizeof(DEVEXT), &KbdNameFull,
1894 FILE_DEVICE_8042_PORT, 0, FALSE, &pPortDevObj);
1895 if (NT_SUCCESS(status))
1896 break;
1897 else
1898 KbdNameFull.Length -= DevNameSuff.Length;
1899 }
1900
1901 if (!NT_SUCCESS(status))
1902 goto fail;
1903
1904 PDEVEXT pDevExt = (PDEVEXT)pPortDevObj->DeviceExtension;
1905 *pDevExt = pInit->DevExt;
1906 pDevExt->pDevObj = pPortDevObj;
1907
1908 ULONG resourceListSize = 0;
1909 PCM_RESOURCE_LIST resources = NULL;
1910 CreateResList(pDevExt, &resources, &resourceListSize);
1911
1912 RtlInitUnicodeString(&resourceDeviceClass, NULL);
1913
1914 resourceDeviceClass.MaximumLength = KbdNameBase.Length + sizeof(L"/") + MouNameBase.Length;
1915 resourceDeviceClass.Buffer = (PWSTR)ExAllocatePool(PagedPool, resourceDeviceClass.MaximumLength);
1916 if (!resourceDeviceClass.Buffer)
1917 {
1918 status = STATUS_UNSUCCESSFUL;
1919 goto fail;
1920 }
1921
1922 RtlZeroMemory(resourceDeviceClass.Buffer, resourceDeviceClass.MaximumLength);
1923 RtlAppendUnicodeStringToString(&resourceDeviceClass, &KbdNameBase);
1924 RtlAppendUnicodeToString(&resourceDeviceClass, L"/");
1925 RtlAppendUnicodeStringToString(&resourceDeviceClass, &MouNameBase);
1926
1927 IoReportResourceUsage(&resourceDeviceClass, pDrvObj, NULL, 0, pPortDevObj,
1928 resources, resourceListSize, FALSE, &fConflict);
1929 if (fConflict)
1930 {
1931 status = STATUS_INSUFFICIENT_RESOURCES;
1932 goto fail;
1933 }
1934
1935 for (unsigned i = 0; i < pDevExt->Cfg.cPorts; i++)
1936 {
1937 addressSpace = (pDevExt->Cfg.aPorts[i].Flags & CM_RESOURCE_PORT_IO) == CM_RESOURCE_PORT_IO ? 1 : 0;
1938 if (!HalTranslateBusAddress(pDevExt->Cfg.InterfaceType,
1939 pDevExt->Cfg.uBusNr,
1940 pDevExt->Cfg.aPorts[i].u.Port.Start,
1941 &addressSpace, &Phys))
1942 {
1943 addressSpace = 1;
1944 Phys.QuadPart = 0;
1945 }
1946
1947 if (!addressSpace)
1948 {
1949 pDevExt->fUnmapRegs = TRUE;
1950 pDevExt->DevRegs[i] = (PUCHAR)MmMapIoSpace(Phys, pDevExt->Cfg.aPorts[i].u.Port.Length,
1951 (MEMORY_CACHING_TYPE)FALSE);
1952 }
1953 else
1954 {
1955 pDevExt->fUnmapRegs = FALSE;
1956 pDevExt->DevRegs[i] = (PUCHAR)Phys.LowPart;
1957 }
1958
1959 if (!pDevExt->DevRegs[i])
1960 {
1961 status = STATUS_NONE_MAPPED;
1962 goto fail;
1963 }
1964 }
1965
1966 pPortDevObj->Flags |= DO_BUFFERED_IO;
1967
1968 InitHw(pPortDevObj);
1969
1970 KeInitializeSpinLock(&pDevExt->ShIntObj);
1971
1972 if (pDevExt->HardwarePresent & KEYBOARD_HARDWARE_PRESENT)
1973 {
1974 pDevExt->KbdExt.InputData = (PKEYBOARD_INPUT_DATA)
1975 ExAllocatePool(NonPagedPool, pDevExt->Cfg.KbdAttr.InputDataQueueLength);
1976
1977 if (!pDevExt->KbdExt.InputData)
1978 {
1979 status = STATUS_INSUFFICIENT_RESOURCES;
1980 goto fail;
1981 }
1982
1983 pDevExt->KbdExt.DataEnd =
1984 (PKEYBOARD_INPUT_DATA)((PCHAR) (pDevExt->KbdExt.InputData) + pDevExt->Cfg.KbdAttr.InputDataQueueLength);
1985
1986 RtlZeroMemory(pDevExt->KbdExt.InputData, pDevExt->Cfg.KbdAttr.InputDataQueueLength);
1987 }
1988
1989 if (pDevExt->HardwarePresent & MOUSE_HARDWARE_PRESENT)
1990 {
1991 RtlInitUnicodeString(&MouNameFull, NULL);
1992
1993 MouNameFull.MaximumLength = sizeof(L"\\Device\\") + MouNameBase.Length + DevNameSuff.MaximumLength;
1994 MouNameFull.Buffer = (PWSTR)ExAllocatePool(PagedPool, MouNameFull.MaximumLength);
1995
1996 if (!MouNameFull.Buffer)
1997 {
1998 status = STATUS_UNSUCCESSFUL;
1999 goto fail;
2000 }
2001
2002 RtlZeroMemory(MouNameFull.Buffer, MouNameFull.MaximumLength);
2003 RtlAppendUnicodeToString(&MouNameFull, L"\\Device\\");
2004 RtlAppendUnicodeToString(&MouNameFull, MouNameBase.Buffer);
2005
2006 RtlZeroMemory(DevNameSuff.Buffer, DevNameSuff.MaximumLength);
2007 DevNameSuff.Length = 0;
2008
2009 for (unsigned i = 0; i < POINTER_PORTS_MAXIMUM; i++)
2010 {
2011 status = RtlIntegerToUnicodeString(i, 10, &DevNameSuff);
2012 if (!NT_SUCCESS(status))
2013 break;
2014
2015 RtlAppendUnicodeStringToString(&MouNameFull, &DevNameSuff);
2016 LogFlow(("VBoxMouseNT::DriverEntry: pointer port name (symbolic link) = %S\n", MouNameFull.Buffer));
2017
2018 status = IoCreateSymbolicLink(&MouNameFull, &KbdNameFull);
2019 if (NT_SUCCESS(status))
2020 break;
2021 else
2022 MouNameFull.Length -= DevNameSuff.Length;
2023 }
2024 if (!NT_SUCCESS(status))
2025 goto fail;
2026
2027 pDevExt->MouExt.InputData =
2028 (PMOUSE_INPUT_DATA)ExAllocatePool(NonPagedPool, pDevExt->Cfg.MouAttr.InputDataQueueLength);
2029 if (!pDevExt->MouExt.InputData)
2030 {
2031 status = STATUS_INSUFFICIENT_RESOURCES;
2032 goto fail;
2033 }
2034
2035 pDevExt->MouExt.DataEnd = (PMOUSE_INPUT_DATA)((PCHAR) (pDevExt->MouExt.InputData) + pDevExt->Cfg.MouAttr.InputDataQueueLength);
2036
2037 RtlZeroMemory(pDevExt->MouExt.InputData, pDevExt->Cfg.MouAttr.InputDataQueueLength);
2038 }
2039
2040 pDevExt->KbdExt.ConnectData.ClassDeviceObject = NULL;
2041 pDevExt->KbdExt.ConnectData.ClassService = NULL;
2042 pDevExt->MouExt.ConnectData.ClassDeviceObject = NULL;
2043 pDevExt->MouExt.ConnectData.ClassService = NULL;
2044
2045 I8042INITDATACTX initDataCtx;
2046 initDataCtx.pDevExt = pDevExt;
2047 initDataCtx.DevType = KbdDevType;
2048 InitDataQueue(&initDataCtx);
2049 initDataCtx.DevType = MouDevType;
2050 InitDataQueue(&initDataCtx);
2051
2052 pDevExt->DpcInterlockKeyboard = -1;
2053 pDevExt->DpcInterlockMouse = -1;
2054
2055 IoInitializeDpcRequest(pPortDevObj, CompleteDpc);
2056 KeInitializeDpc(&pDevExt->RetriesExceededDpc, (PKDEFERRED_ROUTINE)CtrlRetriesExceededDpc, pPortDevObj);
2057 KeInitializeDpc(&pDevExt->KeyboardIsrDpc, (PKDEFERRED_ROUTINE)CtrlKbdIsrDpc, pPortDevObj);
2058 KeInitializeDpc(&pDevExt->KeyboardIsrDpcRetry, (PKDEFERRED_ROUTINE)CtrlKbdIsrDpc, pPortDevObj);
2059 KeInitializeDpc(&pDevExt->MouseIsrDpc, (PKDEFERRED_ROUTINE)CtrlMouIsrDpc, pPortDevObj);
2060 KeInitializeDpc(&pDevExt->MouseIsrDpcRetry, (PKDEFERRED_ROUTINE)CtrlMouIsrDpc, pPortDevObj);
2061 KeInitializeDpc(&pDevExt->TimeOutDpc, (PKDEFERRED_ROUTINE)CtrlTimeoutDpc, pPortDevObj);
2062
2063 KeInitializeTimer(&pDevExt->CommandTimer);
2064 pDevExt->TimerCount = -1;
2065
2066 KeInitializeTimer(&pDevExt->KbdExt.DataConsumptionTimer);
2067 KeInitializeTimer(&pDevExt->MouExt.DataConsumptionTimer);
2068
2069 IntVecKbd = HalGetInterruptVector(pDevExt->Cfg.InterfaceType,
2070 pDevExt->Cfg.uBusNr,
2071 pDevExt->Cfg.KbdInt.u.Interrupt.Level,
2072 pDevExt->Cfg.KbdInt.u.Interrupt.Vector,
2073 &IrqlKbd, &AffKbd);
2074
2075 IntVecMou = HalGetInterruptVector(pDevExt->Cfg.InterfaceType,
2076 pDevExt->Cfg.uBusNr,
2077 pDevExt->Cfg.MouInt.u.Interrupt.Level,
2078 pDevExt->Cfg.MouInt.u.Interrupt.Vector,
2079 &IrqlMou, &AffMou);
2080
2081 if ( (pDevExt->HardwarePresent & (KEYBOARD_HARDWARE_PRESENT | MOUSE_HARDWARE_PRESENT))
2082 == (KEYBOARD_HARDWARE_PRESENT | MOUSE_HARDWARE_PRESENT))
2083 IrqlCoord = IrqlKbd > IrqlMou ? IrqlKbd : IrqlMou;
2084
2085 if (pDevExt->HardwarePresent & MOUSE_HARDWARE_PRESENT)
2086 {
2087 status = IoConnectInterrupt(&pDevExt->MouIntObj, MouIntHandler, pPortDevObj,
2088 &pDevExt->ShIntObj, IntVecMou, IrqlMou,
2089 (KIRQL) ((IrqlCoord == (KIRQL)0) ? IrqlMou : IrqlCoord),
2090 pDevExt->Cfg.MouInt.Flags == CM_RESOURCE_INTERRUPT_LATCHED
2091 ? Latched : LevelSensitive,
2092 pDevExt->Cfg.MouInt.ShareDisposition,
2093 AffMou, pDevExt->Cfg.fFloatSave);
2094 if (!NT_SUCCESS(status))
2095 goto fail;
2096
2097 status = MouEnableTrans(pPortDevObj);
2098 if (!NT_SUCCESS(status))
2099 status = STATUS_SUCCESS;
2100 }
2101
2102 if (pDevExt->HardwarePresent & KEYBOARD_HARDWARE_PRESENT)
2103 {
2104 status = IoConnectInterrupt(&pDevExt->KbdIntObj, KbdIntHandler, pPortDevObj,
2105 &pDevExt->ShIntObj, IntVecKbd, IrqlKbd,
2106 (KIRQL) ((IrqlCoord == (KIRQL)0) ? IrqlKbd : IrqlCoord),
2107 pDevExt->Cfg.KbdInt.Flags == CM_RESOURCE_INTERRUPT_LATCHED
2108 ? Latched : LevelSensitive,
2109 pDevExt->Cfg.KbdInt.ShareDisposition,
2110 AffKbd, pDevExt->Cfg.fFloatSave);
2111 if (!NT_SUCCESS(status))
2112 goto fail;
2113 }
2114
2115 if (pDevExt->HardwarePresent & KEYBOARD_HARDWARE_PRESENT)
2116 {
2117 status = RtlWriteRegistryValue(RTL_REGISTRY_DEVICEMAP,
2118 KbdNameBase.Buffer, KbdNameFull.Buffer,
2119 REG_SZ,
2120 registryPath.Buffer, registryPath.Length);
2121 if (!NT_SUCCESS(status))
2122 goto fail;
2123 }
2124
2125 if (pDevExt->HardwarePresent & MOUSE_HARDWARE_PRESENT)
2126 {
2127 status = RtlWriteRegistryValue(RTL_REGISTRY_DEVICEMAP,
2128 MouNameBase.Buffer, MouNameFull.Buffer,
2129 REG_SZ,
2130 registryPath.Buffer, registryPath.Length);
2131 if (!NT_SUCCESS(status))
2132 goto fail;
2133 }
2134
2135 ASSERT(status == STATUS_SUCCESS);
2136
2137 int rcVBox = VbglInit();
2138 if (RT_FAILURE(rcVBox))
2139 {
2140 Log(("VBoxMouseNT::DriverEntry: could not initialize guest library, rc = %Rrc\n", rcVBox));
2141 /* Continue working in non-VBox mode. */
2142 }
2143 else
2144 {
2145 VMMDevReqMouseStatus *pReq = NULL;
2146
2147 rcVBox = VbglGRAlloc((VMMDevRequestHeader**)&pReq, sizeof(VMMDevReqMouseStatus), VMMDevReq_SetMouseStatus);
2148 if (RT_SUCCESS(rcVBox))
2149 {
2150 /* Inform host that we support absolute */
2151 pReq->mouseFeatures = VMMDEV_MOUSE_GUEST_CAN_ABSOLUTE;
2152 pReq->pointerXPos = 0;
2153 pReq->pointerYPos = 0;
2154 rcVBox = VbglGRPerform(&pReq->header);
2155 if (RT_FAILURE(rcVBox))
2156 Log(("VBoxMouseNT::DriverEntry: ERROR communicating new mouse capabilities to VMMDev. rc = %Rrc\n", rcVBox));
2157 else
2158 {
2159 /* We will use the allocated request buffer in the ServiceCallback to GET mouse status. */
2160 pReq->header.requestType = VMMDevReq_GetMouseStatus;
2161 pDevExt->pReq = pReq;
2162 }
2163 }
2164 else
2165 {
2166 VbglTerminate();
2167 Log(("VBoxMouseNT::DriverEntry: could not allocate request buffer, rc = %Rrc\n", rcVBox));
2168 /* Continue working in non-VBox mode. */
2169 }
2170 }
2171
2172 pDrvObj->DriverStartIo = I8042StartIo;
2173 pDrvObj->MajorFunction[IRP_MJ_CREATE] = I8042OpenClose;
2174 pDrvObj->MajorFunction[IRP_MJ_CLOSE] = I8042OpenClose;
2175 pDrvObj->MajorFunction[IRP_MJ_FLUSH_BUFFERS] = I8042Flush;
2176 pDrvObj->MajorFunction[IRP_MJ_INTERNAL_DEVICE_CONTROL] = I8042DevCtrl;
2177
2178fail:
2179 if (!NT_SUCCESS(status))
2180 {
2181 if (resources)
2182 {
2183 resources->Count = 0;
2184 IoReportResourceUsage(&resourceDeviceClass, pDrvObj, NULL,
2185 0, pPortDevObj, resources, resourceListSize, FALSE, &fConflict);
2186 }
2187
2188 if (pDevExt)
2189 {
2190 if (pDevExt->KbdIntObj)
2191 IoDisconnectInterrupt(pDevExt->KbdIntObj);
2192 if (pDevExt->MouIntObj)
2193 IoDisconnectInterrupt(pDevExt->MouIntObj);
2194 if (pDevExt->KbdExt.InputData)
2195 ExFreePool(pDevExt->KbdExt.InputData);
2196 if (pDevExt->MouExt.InputData)
2197 ExFreePool(pDevExt->MouExt.InputData);
2198 if (pDevExt->fUnmapRegs)
2199 {
2200 for (unsigned i = 0; i < pDevExt->Cfg.cPorts; i++)
2201 if (pDevExt->DevRegs[i])
2202 MmUnmapIoSpace(pDevExt->DevRegs[i], pDevExt->Cfg.aPorts[i].u.Port.Length);
2203 }
2204 }
2205 if (pPortDevObj)
2206 {
2207 if (MouNameFull.Length > 0)
2208 IoDeleteSymbolicLink(&MouNameFull);
2209 IoDeleteDevice(pPortDevObj);
2210 }
2211 }
2212
2213 if (resources)
2214 ExFreePool(resources);
2215 if (pInit)
2216 ExFreePool(pInit);
2217 if (DevNameSuff.MaximumLength)
2218 ExFreePool(DevNameSuff.Buffer);
2219 if (KbdNameFull.MaximumLength)
2220 ExFreePool(KbdNameFull.Buffer);
2221 if (MouNameFull.MaximumLength)
2222 ExFreePool(MouNameFull.Buffer);
2223 if (resourceDeviceClass.MaximumLength)
2224 ExFreePool(resourceDeviceClass.Buffer);
2225 if (registryPath.MaximumLength)
2226 ExFreePool(registryPath.Buffer);
2227
2228 LogFlow(("VBoxMouseNT::DriverEntry: leave, status = %d\n", status));
2229
2230 return status;
2231}
2232
2233static VOID I8042Unload(PDRIVER_OBJECT pDrvObj)
2234{
2235 NOREF(pDrvObj);
2236}
2237
2238/**
2239 * Build a resource list.
2240 */
2241static VOID CreateResList(PDEVEXT pDevExt, PCM_RESOURCE_LIST *pResList, PULONG pResListSize)
2242{
2243 ULONG cPorts = pDevExt->Cfg.cPorts;
2244 if (pDevExt->Cfg.KbdInt.Type == CmResourceTypeInterrupt)
2245 cPorts++;
2246 if (pDevExt->Cfg.MouInt.Type == CmResourceTypeInterrupt)
2247 cPorts++;
2248
2249 *pResListSize = sizeof(CM_RESOURCE_LIST) + ((cPorts - 1) * sizeof(CM_PARTIAL_RESOURCE_DESCRIPTOR));
2250 *pResList = (PCM_RESOURCE_LIST)ExAllocatePool(PagedPool, *pResListSize);
2251 if (!*pResList)
2252 {
2253 *pResListSize = 0;
2254 return;
2255 }
2256
2257 RtlZeroMemory(*pResList, *pResListSize);
2258
2259 (*pResList)->Count = 1;
2260 (*pResList)->List[0].InterfaceType = pDevExt->Cfg.InterfaceType;
2261 (*pResList)->List[0].BusNumber = pDevExt->Cfg.uBusNr;
2262
2263 (*pResList)->List[0].PartialResourceList.Count = cPorts;
2264 ULONG i = 0;
2265 if (pDevExt->Cfg.KbdInt.Type == CmResourceTypeInterrupt)
2266 (*pResList)->List[0].PartialResourceList.PartialDescriptors[i++] = pDevExt->Cfg.KbdInt;
2267 if (pDevExt->Cfg.MouInt.Type == CmResourceTypeInterrupt)
2268 (*pResList)->List[0].PartialResourceList.PartialDescriptors[i++] = pDevExt->Cfg.MouInt;
2269 for (unsigned j = 0; j < pDevExt->Cfg.cPorts; j++)
2270 (*pResList)->List[0].PartialResourceList.PartialDescriptors[i++] = pDevExt->Cfg.aPorts[j];
2271}
2272
2273/**
2274 * Read the i8042 controller command byte
2275 */
2276static NTSTATUS GetCtrlCmd(ULONG HwDisEnMask, PDEVEXT pDevExt, PUCHAR pByte)
2277{
2278 NTSTATUS status;
2279 if (HwDisEnMask & KEYBOARD_HARDWARE_PRESENT)
2280 {
2281 status = PutBytePoll(i8042Cmd, FALSE /*=wait*/, NoDevice, pDevExt, I8042_DISABLE_KEYBOARD_DEVICE);
2282 if (!NT_SUCCESS(status))
2283 return status;
2284 }
2285
2286 if (HwDisEnMask & MOUSE_HARDWARE_PRESENT)
2287 {
2288 status = PutBytePoll(i8042Cmd, FALSE /*=wait*/, NoDevice, pDevExt, I8042_DISABLE_MOUSE_DEVICE);
2289 if (!NT_SUCCESS(status))
2290 {
2291 if (HwDisEnMask & KEYBOARD_HARDWARE_PRESENT)
2292 PutBytePoll(i8042Cmd, FALSE/*=wait*/, NoDevice, pDevExt, I8042_ENABLE_KEYBOARD_DEVICE);
2293 return status;
2294 }
2295 }
2296
2297 status = PutBytePoll(i8042Cmd, FALSE /*=wait*/, NoDevice, pDevExt, I8042_READ_CONTROLLER_COMMAND_BYTE);
2298 if (NT_SUCCESS(status))
2299 {
2300 for (unsigned iRetry = 0; iRetry < 5; iRetry++)
2301 {
2302 status = GetBytePoll(CtrlDevType, pDevExt, pByte);
2303 if (NT_SUCCESS(status))
2304 break;
2305 if (status == STATUS_IO_TIMEOUT)
2306 KeStallExecutionProcessor(50);
2307 else
2308 break;
2309 }
2310 }
2311
2312 NTSTATUS status2;
2313 if (HwDisEnMask & KEYBOARD_HARDWARE_PRESENT)
2314 {
2315 status2 = PutBytePoll(i8042Cmd, FALSE /*=wait*/, NoDevice, pDevExt, I8042_ENABLE_KEYBOARD_DEVICE);
2316 if (!NT_SUCCESS(status2))
2317 {
2318 if (NT_SUCCESS(status))
2319 status = status2;
2320 }
2321 else if (status == STATUS_SUCCESS)
2322 *pByte &= (UCHAR)~CCB_DISABLE_KEYBOARD_DEVICE;
2323 }
2324
2325 if (HwDisEnMask & MOUSE_HARDWARE_PRESENT)
2326 {
2327 status2 = PutBytePoll(i8042Cmd, FALSE /*=wait*/, NoDevice, pDevExt, I8042_ENABLE_MOUSE_DEVICE);
2328 if (!NT_SUCCESS(status2))
2329 {
2330 if (NT_SUCCESS(status))
2331 status = status2;
2332 }
2333 else if (NT_SUCCESS(status))
2334 *pByte &= (UCHAR)~CCB_DISABLE_MOUSE_DEVICE;
2335 }
2336 return status;
2337}
2338
2339/**
2340 * Write the i8042 controller command byte.
2341 */
2342static NTSTATUS PutCtrlCmd(PDEVEXT pDevExt, UCHAR Byte)
2343{
2344 NTSTATUS status;
2345 status = PutBytePoll(i8042Cmd, FALSE /*=wait*/, NoDevice, pDevExt, I8042_WRITE_CONTROLLER_COMMAND_BYTE);
2346 if (!NT_SUCCESS(status))
2347 return status;
2348
2349 return (PutBytePoll(i8042Dat, FALSE /*=wait*/, NoDevice, pDevExt, Byte));
2350}
2351
2352/**
2353 * Read the i8042 controller command byte.
2354 */
2355static VOID TransCtrlCmd(PDEVEXT pDevExt, PI8042TRANSMITCCBCTX pCtx)
2356{
2357 UCHAR bCtrlCmd;
2358 pCtx->Status = GetCtrlCmd(pCtx->HwDisEnMask, pDevExt, &bCtrlCmd);
2359 if (!NT_SUCCESS(pCtx->Status))
2360 return;
2361
2362 if (pCtx->fAndOp)
2363 bCtrlCmd &= pCtx->ByteMask;
2364 else
2365 bCtrlCmd |= pCtx->ByteMask;
2366
2367 pCtx->Status = PutCtrlCmd(pDevExt, bCtrlCmd);
2368
2369 UCHAR bVrfyCmd;
2370 pCtx->Status = GetCtrlCmd(pCtx->HwDisEnMask, pDevExt, &bVrfyCmd);
2371
2372 if ( NT_SUCCESS(pCtx->Status)
2373 && bVrfyCmd != bCtrlCmd)
2374 pCtx->Status = STATUS_DEVICE_DATA_ERROR;
2375}
2376
2377/**
2378 * Detect the number of mouse buttons.
2379 */
2380static NTSTATUS MouQueryButtons(PDEVICE_OBJECT pDevObj, PUCHAR pNumButtons)
2381{
2382 PDEVEXT pDevExt = (PDEVEXT)pDevObj->DeviceExtension;
2383
2384 NTSTATUS status = PutBytePoll(i8042Dat, TRUE /*=wait*/, MouDevType, pDevExt, SET_MOUSE_RESOLUTION);
2385 if (!NT_SUCCESS(status))
2386 return status;
2387
2388 status = PutBytePoll(i8042Dat, TRUE /*=wait*/, MouDevType, pDevExt, 0x00);
2389 if (!NT_SUCCESS(status))
2390 return status;
2391
2392 for (unsigned i = 0; i < 3; i++)
2393 {
2394 status = PutBytePoll(i8042Dat, TRUE /*=wait*/, MouDevType, pDevExt, SET_MOUSE_SCALING_1TO1);
2395 if (!NT_SUCCESS(status))
2396 return status;
2397 }
2398
2399 status = PutBytePoll(i8042Dat, TRUE /*=wait*/, MouDevType, pDevExt, READ_MOUSE_STATUS);
2400 if (!NT_SUCCESS(status))
2401 return status;
2402 UCHAR byte;
2403 status = GetBytePoll(CtrlDevType, pDevExt, &byte);
2404 if (!NT_SUCCESS(status))
2405 return status;
2406 UCHAR buttons;
2407 status = GetBytePoll(CtrlDevType, pDevExt, &buttons);
2408 if (!NT_SUCCESS(status))
2409 return status;
2410 status = GetBytePoll(CtrlDevType, pDevExt, &byte);
2411 if (!NT_SUCCESS(status))
2412 return status;
2413
2414 if (buttons == 2 || buttons == 3)
2415 *pNumButtons = buttons;
2416 else
2417 *pNumButtons = 0;
2418
2419 return status;
2420}
2421
2422/**
2423 * Initialize the i8042 mouse hardware.
2424 */
2425static NTSTATUS MouInitHw(PDEVICE_OBJECT pDevObj)
2426{
2427 PDEVEXT pDevExt = (PDEVEXT) pDevObj->DeviceExtension;
2428
2429 NTSTATUS status = PutBytePoll(i8042Dat, TRUE /*=wait*/, MouDevType, pDevExt, MOUSE_RESET);
2430 if (!NT_SUCCESS(status))
2431 goto fail;
2432
2433 UCHAR byte;
2434 for (unsigned i = 0; i < 11200; i++)
2435 {
2436 status = GetBytePoll(CtrlDevType, pDevExt, &byte);
2437 if (NT_SUCCESS(status) && byte == (UCHAR) MOUSE_COMPLETE)
2438 break;
2439 if (status != STATUS_IO_TIMEOUT)
2440 break;
2441 KeStallExecutionProcessor(50);
2442 }
2443
2444 if (!NT_SUCCESS(status))
2445 goto fail;
2446
2447 status = GetBytePoll(CtrlDevType, pDevExt, &byte);
2448 if ((!NT_SUCCESS(status)) || (byte != MOUSE_ID_BYTE))
2449 goto fail;
2450
2451 MouFindWheel(pDevObj);
2452
2453 UCHAR numButtons;
2454 status = MouQueryButtons(pDevObj, &numButtons);
2455 if (!NT_SUCCESS(status))
2456 goto fail;
2457 else if (numButtons)
2458 pDevExt->Cfg.MouAttr.NumberOfButtons = numButtons;
2459
2460 status = PutBytePoll(i8042Dat, TRUE /*=wait*/, MouDevType, pDevExt, SET_MOUSE_SAMPLING_RATE);
2461 if (!NT_SUCCESS(status))
2462 goto fail;
2463
2464 status = PutBytePoll(i8042Dat, TRUE /*=wait*/, MouDevType, pDevExt, 60);
2465 if (!NT_SUCCESS(status))
2466 goto fail;
2467
2468 status = PutBytePoll(i8042Dat, TRUE /*=wait*/, MouDevType, pDevExt, SET_MOUSE_RESOLUTION);
2469 if (!NT_SUCCESS(status))
2470 goto fail;
2471
2472 status = PutBytePoll(i8042Dat, TRUE /*=wait*/, MouDevType, pDevExt, (UCHAR)pDevExt->Cfg.MouseResolution);
2473
2474fail:
2475 pDevExt->MouExt.uPrevSignAndOverflow = 0;
2476 pDevExt->MouExt.InputState = MouseExpectingACK;
2477 pDevExt->MouExt.LastByteReceived = 0;
2478
2479 return status;
2480}
2481
2482/**
2483 * Initialize the i8042 keyboard hardware.
2484 */
2485static NTSTATUS KbdInitHw(PDEVICE_OBJECT pDevObj)
2486{
2487 NTSTATUS status;
2488 BOOLEAN fWaitForAck = TRUE;
2489 PDEVEXT pDevExt = (PDEVEXT)pDevObj->DeviceExtension;
2490
2491retry:
2492 PutBytePoll(i8042Dat, fWaitForAck, KbdDevType, pDevExt, KEYBOARD_RESET);
2493
2494 LARGE_INTEGER startOfSpin;
2495 KeQueryTickCount(&startOfSpin);
2496 for (unsigned i = 0; i < 11200; i++)
2497 {
2498 UCHAR byte;
2499 status = GetBytePoll(KbdDevType, pDevExt, &byte);
2500 if (NT_SUCCESS(status))
2501 break;
2502 else
2503 {
2504 if (status == STATUS_IO_TIMEOUT)
2505 {
2506 LARGE_INTEGER nextQuery, difference, tenSeconds;
2507 KeStallExecutionProcessor(50);
2508 KeQueryTickCount(&nextQuery);
2509 difference.QuadPart = nextQuery.QuadPart - startOfSpin.QuadPart;
2510 tenSeconds.QuadPart = 10*10*1000*1000;
2511 ASSERT(KeQueryTimeIncrement() <= MAXLONG);
2512 if (difference.QuadPart*KeQueryTimeIncrement() >= tenSeconds.QuadPart)
2513 break;
2514 }
2515 else
2516 break;
2517 }
2518 }
2519
2520 if (!NT_SUCCESS(status))
2521 {
2522 if (fWaitForAck)
2523 {
2524 fWaitForAck = FALSE;
2525 goto retry;
2526 }
2527 goto fail;
2528 }
2529
2530 I8042TRANSMITCCBCTX Ctx;
2531 Ctx.HwDisEnMask = 0;
2532 Ctx.fAndOp = TRUE;
2533 Ctx.ByteMask = (UCHAR) ~((UCHAR)CCB_KEYBOARD_TRANSLATE_MODE);
2534
2535 TransCtrlCmd(pDevExt, &Ctx);
2536 if (!NT_SUCCESS(Ctx.Status))
2537 TransCtrlCmd(pDevExt, &Ctx);
2538 if (!NT_SUCCESS(Ctx.Status))
2539 {
2540 status = Ctx.Status;
2541 goto fail;
2542 }
2543
2544 PKEYBOARD_ID pId = &pDevExt->Cfg.KbdAttr.KeyboardIdentifier;
2545 status = PutBytePoll(i8042Dat, TRUE /*=wait*/, KbdDevType, pDevExt, SET_KEYBOARD_TYPEMATIC);
2546 if (status == STATUS_SUCCESS)
2547 {
2548 status = PutBytePoll(i8042Dat, TRUE /*=wait*/, KbdDevType, pDevExt,
2549 ConvertTypematic(pDevExt->Cfg.KeyRepeatCurrent.Rate,
2550 pDevExt->Cfg.KeyRepeatCurrent.Delay));
2551 /* ignore errors */
2552 }
2553
2554 status = PutBytePoll(i8042Dat, TRUE /*=wait*/, KbdDevType, pDevExt, SET_KEYBOARD_INDICATORS);
2555 if (status == STATUS_SUCCESS)
2556 {
2557 status = PutBytePoll(i8042Dat, TRUE /*=wait*/, KbdDevType, pDevExt,
2558 (UCHAR)pDevExt->Cfg.KbdInd.LedFlags);
2559 /* ignore errors */
2560 }
2561 status = STATUS_SUCCESS;
2562
2563 if (pDevExt->Cfg.KbdAttr.KeyboardMode == 1)
2564 {
2565 Ctx.HwDisEnMask = 0;
2566 Ctx.fAndOp = FALSE;
2567 Ctx.ByteMask = CCB_KEYBOARD_TRANSLATE_MODE;
2568 TransCtrlCmd(pDevExt, &Ctx);
2569 if (!NT_SUCCESS(Ctx.Status))
2570 {
2571 if (Ctx.Status == STATUS_DEVICE_DATA_ERROR)
2572 {
2573 if (ENHANCED_KEYBOARD(*pId))
2574 {
2575 status = PutBytePoll(i8042Dat, TRUE /*=wait*/, KbdDevType, pDevExt, SELECT_SCAN_CODE_SET);
2576 if (!NT_SUCCESS(status))
2577 pDevExt->Cfg.KbdAttr.KeyboardMode = 2;
2578 else
2579 {
2580 status = PutBytePoll(i8042Dat, TRUE /*=wait*/, KbdDevType, pDevExt, 1);
2581 if (!NT_SUCCESS(status))
2582 pDevExt->Cfg.KbdAttr.KeyboardMode = 2;
2583 }
2584 }
2585 }
2586 else
2587 {
2588 status = Ctx.Status;
2589 goto fail;
2590 }
2591 }
2592 }
2593
2594fail:
2595 pDevExt->KbdExt.CurrentOutput.State = Idle;
2596 pDevExt->KbdExt.CurrentOutput.FirstByte = 0;
2597 pDevExt->KbdExt.CurrentOutput.LastByte = 0;
2598
2599 return status;
2600}
2601
2602/**
2603 * Initialize the i8042 controller, keyboard and mouse.
2604 */
2605static VOID InitHw(PDEVICE_OBJECT pDevObj)
2606{
2607 PDEVEXT pDevExt = (PDEVEXT) pDevObj->DeviceExtension;
2608 PUCHAR dataAddress = pDevExt->DevRegs[i8042Dat];
2609 PUCHAR commandAddress = pDevExt->DevRegs[i8042Cmd];
2610
2611 DrainOutBuf(dataAddress, commandAddress);
2612
2613 I8042TRANSMITCCBCTX Ctx;
2614 Ctx.HwDisEnMask = 0;
2615 Ctx.fAndOp = TRUE;
2616 Ctx.ByteMask = (UCHAR) ~((UCHAR)CCB_ENABLE_KEYBOARD_INTERRUPT | (UCHAR)CCB_ENABLE_MOUSE_INTERRUPT);
2617 TransCtrlCmd(pDevExt, &Ctx);
2618 if (!NT_SUCCESS(Ctx.Status))
2619 return;
2620
2621 DrainOutBuf(dataAddress, commandAddress);
2622
2623 if (pDevExt->HardwarePresent & MOUSE_HARDWARE_PRESENT)
2624 {
2625 NTSTATUS status = MouInitHw(pDevObj);
2626 if (!NT_SUCCESS(status))
2627 pDevExt->HardwarePresent &= ~MOUSE_HARDWARE_PRESENT;
2628 }
2629
2630 if (pDevExt->HardwarePresent & KEYBOARD_HARDWARE_PRESENT)
2631 {
2632 NTSTATUS status = KbdInitHw(pDevObj);
2633 if (!NT_SUCCESS(status))
2634 pDevExt->HardwarePresent &= ~KEYBOARD_HARDWARE_PRESENT;
2635 }
2636
2637 if (pDevExt->HardwarePresent & KEYBOARD_HARDWARE_PRESENT)
2638 {
2639 NTSTATUS status = PutBytePoll(i8042Cmd, FALSE /*=wait*/, NoDevice, pDevExt, I8042_ENABLE_KEYBOARD_DEVICE);
2640 if (!NT_SUCCESS(status))
2641 pDevExt->HardwarePresent &= ~KEYBOARD_HARDWARE_PRESENT;
2642
2643 DrainOutBuf(dataAddress, commandAddress);
2644 }
2645
2646 if (pDevExt->HardwarePresent & MOUSE_HARDWARE_PRESENT)
2647 {
2648 NTSTATUS status = PutBytePoll(i8042Cmd, FALSE /*=wait*/, NoDevice, pDevExt, I8042_ENABLE_MOUSE_DEVICE);
2649 if (!NT_SUCCESS(status))
2650 pDevExt->HardwarePresent &= ~MOUSE_HARDWARE_PRESENT;
2651 DrainOutBuf(dataAddress, commandAddress);
2652 }
2653
2654 if (pDevExt->HardwarePresent)
2655 {
2656 Ctx.HwDisEnMask = pDevExt->HardwarePresent;
2657 Ctx.fAndOp = FALSE;
2658 Ctx.ByteMask = (pDevExt->HardwarePresent & KEYBOARD_HARDWARE_PRESENT)
2659 ? CCB_ENABLE_KEYBOARD_INTERRUPT : 0;
2660 Ctx.ByteMask |= (UCHAR)
2661 (pDevExt->HardwarePresent & MOUSE_HARDWARE_PRESENT) ? CCB_ENABLE_MOUSE_INTERRUPT : 0;
2662 TransCtrlCmd(pDevExt, &Ctx);
2663 if (!NT_SUCCESS(Ctx.Status))
2664 {
2665 /* ignore */
2666 }
2667 }
2668}
2669
2670/**
2671 * retrieve the drivers service parameters from the registry
2672 */
2673static VOID HwGetRegstry(PINITEXT pInit, PUNICODE_STRING RegistryPath,
2674 PUNICODE_STRING KeyboardDeviceName, PUNICODE_STRING PointerDeviceName)
2675
2676{
2677 PRTL_QUERY_REGISTRY_TABLE aQuery = NULL;
2678 UNICODE_STRING parametersPath;
2679 UNICODE_STRING defaultPointerName;
2680 UNICODE_STRING defaultKeyboardName;
2681 USHORT defaultResendIterations = 3;
2682 ULONG iResend = 0;
2683 USHORT defaultPollingIterations = 12000;
2684 ULONG pollingIterations = 0;
2685 USHORT defaultPollingIterationsMaximum = 12000;
2686 ULONG pollingIterationsMaximum = 0;
2687 USHORT defaultPollStatusIterations = 12000;
2688 ULONG pollStatusIterations = 0;
2689 ULONG defaultDataQueueSize = 100;
2690 ULONG cButtons = 2;
2691 USHORT cButtonsDef = 2;
2692 ULONG sampleRate = 60;
2693 USHORT defaultSampleRate = 60;
2694 ULONG mouseResolution = 3;
2695 USHORT defaultMouseResolution = 3;
2696 ULONG overrideKeyboardType = 0;
2697 ULONG invalidKeyboardType = 0;
2698 ULONG overrideKeyboardSubtype = (ULONG)-1;
2699 ULONG invalidKeyboardSubtype = (ULONG)-1;
2700 ULONG defaultSynchPacket100ns = 10000000UL;
2701 ULONG enableWheelDetection = 0;
2702 ULONG defaultEnableWheelDetection = 1;
2703 PWSTR path = NULL;
2704 USHORT queries = 15;
2705 PI8042CFGINF pCfg = &pInit->DevExt.Cfg;
2706 NTSTATUS status = STATUS_SUCCESS;
2707
2708 pCfg->StallMicroseconds = 50;
2709 parametersPath.Buffer = NULL;
2710
2711 path = RegistryPath->Buffer;
2712 if (NT_SUCCESS(status))
2713 {
2714 aQuery = (PRTL_QUERY_REGISTRY_TABLE)
2715 ExAllocatePool(PagedPool, sizeof(RTL_QUERY_REGISTRY_TABLE) * (queries + 1));
2716 if (!aQuery)
2717 status = STATUS_UNSUCCESSFUL;
2718 else
2719 {
2720 RtlZeroMemory(aQuery, sizeof(RTL_QUERY_REGISTRY_TABLE) * (queries + 1));
2721 RtlInitUnicodeString(&parametersPath, NULL);
2722 parametersPath.MaximumLength = RegistryPath->Length + sizeof(L"\\Parameters");
2723 parametersPath.Buffer = (PWSTR)ExAllocatePool(PagedPool, parametersPath.MaximumLength);
2724 if (!parametersPath.Buffer)
2725 status = STATUS_UNSUCCESSFUL;
2726 }
2727 }
2728
2729 if (NT_SUCCESS(status))
2730 {
2731 RtlZeroMemory(parametersPath.Buffer, parametersPath.MaximumLength);
2732 RtlAppendUnicodeToString(&parametersPath, path);
2733 RtlAppendUnicodeToString(&parametersPath, L"\\Parameters");
2734
2735 RtlInitUnicodeString(&defaultKeyboardName, L"KeyboardPort");
2736 RtlInitUnicodeString(&defaultPointerName, L"PointerPort");
2737
2738 aQuery[0].Flags = RTL_QUERY_REGISTRY_DIRECT;
2739 aQuery[0].Name = L"iResend";
2740 aQuery[0].EntryContext = &iResend;
2741 aQuery[0].DefaultType = REG_DWORD;
2742 aQuery[0].DefaultData = &defaultResendIterations;
2743 aQuery[0].DefaultLength = sizeof(USHORT);
2744
2745 aQuery[1].Flags = RTL_QUERY_REGISTRY_DIRECT;
2746 aQuery[1].Name = L"PollingIterations";
2747 aQuery[1].EntryContext = &pollingIterations;
2748 aQuery[1].DefaultType = REG_DWORD;
2749 aQuery[1].DefaultData = &defaultPollingIterations;
2750 aQuery[1].DefaultLength = sizeof(USHORT);
2751
2752 aQuery[2].Flags = RTL_QUERY_REGISTRY_DIRECT;
2753 aQuery[2].Name = L"PollingIterationsMaximum";
2754 aQuery[2].EntryContext = &pollingIterationsMaximum;
2755 aQuery[2].DefaultType = REG_DWORD;
2756 aQuery[2].DefaultData = &defaultPollingIterationsMaximum;
2757 aQuery[2].DefaultLength = sizeof(USHORT);
2758
2759 aQuery[3].Flags = RTL_QUERY_REGISTRY_DIRECT;
2760 aQuery[3].Name = L"KeyboardDataQueueSize";
2761 aQuery[3].EntryContext = &pCfg->KbdAttr.InputDataQueueLength;
2762 aQuery[3].DefaultType = REG_DWORD;
2763 aQuery[3].DefaultData = &defaultDataQueueSize;
2764 aQuery[3].DefaultLength = sizeof(ULONG);
2765
2766 aQuery[4].Flags = RTL_QUERY_REGISTRY_DIRECT;
2767 aQuery[4].Name = L"MouseDataQueueSize";
2768 aQuery[4].EntryContext = &pCfg->MouAttr.InputDataQueueLength;
2769 aQuery[4].DefaultType = REG_DWORD;
2770 aQuery[4].DefaultData = &defaultDataQueueSize;
2771 aQuery[4].DefaultLength = sizeof(ULONG);
2772
2773 aQuery[5].Flags = RTL_QUERY_REGISTRY_DIRECT;
2774 aQuery[5].Name = L"NumberOfButtons";
2775 aQuery[5].EntryContext = &cButtons;
2776 aQuery[5].DefaultType = REG_DWORD;
2777 aQuery[5].DefaultData = &cButtonsDef;
2778 aQuery[5].DefaultLength = sizeof(USHORT);
2779
2780 aQuery[6].Flags = RTL_QUERY_REGISTRY_DIRECT;
2781 aQuery[6].Name = L"SampleRate";
2782 aQuery[6].EntryContext = &sampleRate;
2783 aQuery[6].DefaultType = REG_DWORD;
2784 aQuery[6].DefaultData = &defaultSampleRate;
2785 aQuery[6].DefaultLength = sizeof(USHORT);
2786
2787 aQuery[7].Flags = RTL_QUERY_REGISTRY_DIRECT;
2788 aQuery[7].Name = L"MouseResolution";
2789 aQuery[7].EntryContext = &mouseResolution;
2790 aQuery[7].DefaultType = REG_DWORD;
2791 aQuery[7].DefaultData = &defaultMouseResolution;
2792 aQuery[7].DefaultLength = sizeof(USHORT);
2793
2794 aQuery[8].Flags = RTL_QUERY_REGISTRY_DIRECT;
2795 aQuery[8].Name = L"OverrideKeyboardType";
2796 aQuery[8].EntryContext = &overrideKeyboardType;
2797 aQuery[8].DefaultType = REG_DWORD;
2798 aQuery[8].DefaultData = &invalidKeyboardType;
2799 aQuery[8].DefaultLength = sizeof(ULONG);
2800
2801 aQuery[9].Flags = RTL_QUERY_REGISTRY_DIRECT;
2802 aQuery[9].Name = L"OverrideKeyboardSubtype";
2803 aQuery[9].EntryContext = &overrideKeyboardSubtype;
2804 aQuery[9].DefaultType = REG_DWORD;
2805 aQuery[9].DefaultData = &invalidKeyboardSubtype;
2806 aQuery[9].DefaultLength = sizeof(ULONG);
2807
2808 aQuery[10].Flags = RTL_QUERY_REGISTRY_DIRECT;
2809 aQuery[10].Name = L"KeyboardDeviceBaseName";
2810 aQuery[10].EntryContext = KeyboardDeviceName;
2811 aQuery[10].DefaultType = REG_SZ;
2812 aQuery[10].DefaultData = defaultKeyboardName.Buffer;
2813 aQuery[10].DefaultLength = 0;
2814
2815 aQuery[11].Flags = RTL_QUERY_REGISTRY_DIRECT;
2816 aQuery[11].Name = L"PointerDeviceBaseName";
2817 aQuery[11].EntryContext = PointerDeviceName;
2818 aQuery[11].DefaultType = REG_SZ;
2819 aQuery[11].DefaultData = defaultPointerName.Buffer;
2820 aQuery[11].DefaultLength = 0;
2821
2822 aQuery[12].Flags = RTL_QUERY_REGISTRY_DIRECT;
2823 aQuery[12].Name = L"MouseSynchIn100ns";
2824 aQuery[12].EntryContext = &pInit->DevExt.MouExt.SynchTickCount;
2825 aQuery[12].DefaultType = REG_DWORD;
2826 aQuery[12].DefaultData = &defaultSynchPacket100ns;
2827 aQuery[12].DefaultLength = sizeof(ULONG);
2828
2829 aQuery[13].Flags = RTL_QUERY_REGISTRY_DIRECT;
2830 aQuery[13].Name = L"PollStatusIterations";
2831 aQuery[13].EntryContext = &pollStatusIterations;
2832 aQuery[13].DefaultType = REG_DWORD;
2833 aQuery[13].DefaultData = &defaultPollStatusIterations;
2834 aQuery[13].DefaultLength = sizeof(USHORT);
2835
2836 aQuery[14].Flags = RTL_QUERY_REGISTRY_DIRECT;
2837 aQuery[14].Name = L"EnableWheelDetection";
2838 aQuery[14].EntryContext = &enableWheelDetection;
2839 aQuery[14].DefaultType = REG_DWORD;
2840 aQuery[14].DefaultData = &defaultEnableWheelDetection;
2841 aQuery[14].DefaultLength = sizeof(ULONG);
2842
2843 status = RtlQueryRegistryValues(RTL_REGISTRY_ABSOLUTE | RTL_REGISTRY_OPTIONAL,
2844 parametersPath.Buffer, aQuery, NULL, NULL);
2845 }
2846
2847 if (!NT_SUCCESS(status))
2848 {
2849 /* driver defaults */
2850 pCfg->iResend = defaultResendIterations;
2851 pCfg->PollingIterations = defaultPollingIterations;
2852 pCfg->PollingIterationsMaximum = defaultPollingIterationsMaximum;
2853 pCfg->PollStatusIterations = defaultPollStatusIterations;
2854 pCfg->KbdAttr.InputDataQueueLength = defaultDataQueueSize;
2855 pCfg->MouAttr.InputDataQueueLength = defaultDataQueueSize;
2856 pCfg->EnableWheelDetection = defaultEnableWheelDetection;
2857 pInit->DevExt.MouExt.SynchTickCount = defaultSynchPacket100ns;
2858 RtlCopyUnicodeString(KeyboardDeviceName, &defaultKeyboardName);
2859 RtlCopyUnicodeString(PointerDeviceName, &defaultPointerName);
2860 }
2861 else
2862 {
2863 pCfg->iResend = (USHORT)iResend;
2864 pCfg->PollingIterations = (USHORT) pollingIterations;
2865 pCfg->PollingIterationsMaximum = (USHORT) pollingIterationsMaximum;
2866 pCfg->PollStatusIterations = (USHORT) pollStatusIterations;
2867 pCfg->EnableWheelDetection = (ULONG) ((enableWheelDetection) ? 1 : 0);
2868 }
2869
2870 if (pCfg->KbdAttr.InputDataQueueLength == 0)
2871 pCfg->KbdAttr.InputDataQueueLength = defaultDataQueueSize;
2872 pCfg->KbdAttr.InputDataQueueLength *= sizeof(KEYBOARD_INPUT_DATA);
2873
2874 if (pCfg->MouAttr.InputDataQueueLength == 0)
2875 pCfg->MouAttr.InputDataQueueLength = defaultDataQueueSize;
2876 pCfg->MouAttr.InputDataQueueLength *= sizeof(MOUSE_INPUT_DATA);
2877
2878 pCfg->MouAttr.NumberOfButtons = (USHORT)cButtons;
2879 pCfg->MouAttr.SampleRate = (USHORT)sampleRate;
2880 pCfg->MouseResolution = (USHORT)mouseResolution;
2881
2882 if (overrideKeyboardType != invalidKeyboardType)
2883 {
2884 if (overrideKeyboardType <= RT_ELEMENTS(s_aKeybType))
2885 pCfg->KbdAttr.KeyboardIdentifier.Type = (UCHAR) overrideKeyboardType;
2886 }
2887
2888 if (overrideKeyboardSubtype != invalidKeyboardSubtype)
2889 pCfg->KbdAttr.KeyboardIdentifier.Subtype = (UCHAR) overrideKeyboardSubtype;
2890
2891 if (pInit->DevExt.MouExt.SynchTickCount == 0)
2892 pInit->DevExt.MouExt.SynchTickCount = defaultSynchPacket100ns;
2893
2894 pInit->DevExt.MouExt.SynchTickCount /= KeQueryTimeIncrement();
2895
2896 if (parametersPath.Buffer)
2897 ExFreePool(parametersPath.Buffer);
2898 if (aQuery)
2899 ExFreePool(aQuery);
2900}
2901
2902static void GetDevIdentifier(PKEY_VALUE_FULL_INFORMATION *ppInf, PUNICODE_STRING pStr)
2903{
2904 pStr->Length = (USHORT)(*(ppInf + IoQueryDeviceIdentifier))->DataLength;
2905 if (!pStr->Length)
2906 return;
2907 pStr->MaximumLength = pStr->Length;
2908 pStr->Buffer = (PWSTR) (((PUCHAR)(*(ppInf + IoQueryDeviceIdentifier)))
2909 + (*(ppInf + IoQueryDeviceIdentifier))->DataOffset);
2910}
2911
2912static ULONG GetDevCfgData(PKEY_VALUE_FULL_INFORMATION *ppInf, PCM_PARTIAL_RESOURCE_LIST *ppData)
2913{
2914 ULONG DataLength = (*(ppInf + IoQueryDeviceConfigurationData))->DataLength;
2915 if (DataLength)
2916 *ppData = (PCM_PARTIAL_RESOURCE_LIST)( ((PUCHAR) (*(ppInf + IoQueryDeviceConfigurationData)))
2917 + (*(ppInf + IoQueryDeviceConfigurationData))->DataOffset
2918 + FIELD_OFFSET(CM_FULL_RESOURCE_DESCRIPTOR, PartialResourceList));
2919 return DataLength;
2920}
2921
2922/**
2923 * Callout routine. Grab keyboard controller and peripheral configuration
2924 * information.
2925 */
2926static NTSTATUS KbdCallOut(PVOID pCtx, PUNICODE_STRING PathName,
2927 INTERFACE_TYPE BusType, ULONG uBusNr, PKEY_VALUE_FULL_INFORMATION *pBusInf,
2928 CONFIGURATION_TYPE uCtrlType, ULONG uCtrlNr, PKEY_VALUE_FULL_INFORMATION *pCtrlInf,
2929 CONFIGURATION_TYPE uPrfType, ULONG uPrfNr, PKEY_VALUE_FULL_INFORMATION *pPrfInf)
2930{
2931 UNICODE_STRING unicodeIdentifier;
2932 GetDevIdentifier(pPrfInf, &unicodeIdentifier);
2933
2934 PINITEXT pInit = (PINITEXT)pCtx;
2935 PDEVEXT pDevExt = &pInit->DevExt;
2936 if ( (pDevExt->HardwarePresent & KEYBOARD_HARDWARE_PRESENT)
2937 || !unicodeIdentifier.Length)
2938 return STATUS_SUCCESS;
2939
2940 pDevExt->HardwarePresent |= KEYBOARD_HARDWARE_PRESENT;
2941
2942 PI8042CFGINF pCfg = &pDevExt->Cfg;
2943 pCfg->KbdAttr.KeyboardIdentifier.Type = 0;
2944 pCfg->KbdAttr.KeyboardIdentifier.Subtype = 0;
2945
2946 PCM_PARTIAL_RESOURCE_LIST pPrfData;
2947 if (GetDevCfgData(pPrfInf, &pPrfData))
2948 {
2949 unsigned cList = pPrfData->Count;
2950 PCM_PARTIAL_RESOURCE_DESCRIPTOR pResDesc = pPrfData->PartialDescriptors;
2951 for (unsigned i = 0; i < cList; i++, pResDesc++)
2952 {
2953 switch (pResDesc->Type)
2954 {
2955 case CmResourceTypeDeviceSpecific:
2956 {
2957 PCM_KEYBOARD_DEVICE_DATA KbdData = (PCM_KEYBOARD_DEVICE_DATA)(((PUCHAR)pResDesc)
2958 + sizeof(CM_PARTIAL_RESOURCE_DESCRIPTOR));
2959 if (KbdData->Type <= RT_ELEMENTS(s_aKeybType))
2960 pCfg->KbdAttr.KeyboardIdentifier.Type = KbdData->Type;
2961 pCfg->KbdAttr.KeyboardIdentifier.Subtype = KbdData->Subtype;
2962 pCfg->KbdInd.LedFlags = (KbdData->KeyboardFlags >> 4) & 7;
2963 break;
2964 }
2965 default:
2966 break;
2967 }
2968 }
2969 }
2970
2971 if (pCfg->KbdAttr.KeyboardIdentifier.Type == 0)
2972 {
2973 pCfg->KbdAttr.KeyboardIdentifier.Type = 4;
2974 pCfg->KbdInd.LedFlags = 0;
2975 }
2976
2977 pCfg->InterfaceType = BusType;
2978 pCfg->uBusNr = uBusNr;
2979 pCfg->fFloatSave = FALSE;
2980
2981 BOOLEAN fDefIntShare;
2982 KINTERRUPT_MODE DefIntMode;
2983 if (BusType == MicroChannel)
2984 {
2985 fDefIntShare = TRUE;
2986 DefIntMode = LevelSensitive;
2987 }
2988 else
2989 {
2990 fDefIntShare = FALSE;
2991 DefIntMode = Latched;
2992 }
2993
2994 PCM_PARTIAL_RESOURCE_LIST pCtrlData;
2995 if (GetDevCfgData(pCtrlInf, &pCtrlData))
2996 {
2997 unsigned cList = pCtrlData->Count;
2998 PCM_PARTIAL_RESOURCE_DESCRIPTOR pResDesc = pCtrlData->PartialDescriptors;
2999 for (unsigned i = 0; i < cList; i++, pResDesc++)
3000 {
3001 switch (pResDesc->Type)
3002 {
3003 case CmResourceTypePort:
3004 ASSERT(pCfg->cPorts < i8042MaxPorts);
3005 pCfg->aPorts[pCfg->cPorts] = *pResDesc;
3006 pCfg->aPorts[pCfg->cPorts].ShareDisposition = CmResourceShareDriverExclusive;
3007 pCfg->cPorts++;
3008 break;
3009
3010 case CmResourceTypeInterrupt:
3011 pCfg->KbdInt = *pResDesc;
3012 pCfg->KbdInt.ShareDisposition = fDefIntShare ? CmResourceShareShared
3013 : CmResourceShareDeviceExclusive;
3014 break;
3015
3016 case CmResourceTypeDeviceSpecific:
3017 break;
3018
3019 default:
3020 break;
3021 }
3022 }
3023 }
3024
3025 if (!(pCfg->KbdInt.Type & CmResourceTypeInterrupt))
3026 {
3027 pCfg->KbdInt.Type = CmResourceTypeInterrupt;
3028 pCfg->KbdInt.ShareDisposition = fDefIntShare ? CmResourceShareShared
3029 : CmResourceShareDeviceExclusive;
3030 pCfg->KbdInt.Flags = (DefIntMode == Latched) ? CM_RESOURCE_INTERRUPT_LATCHED
3031 : CM_RESOURCE_INTERRUPT_LEVEL_SENSITIVE;
3032 pCfg->KbdInt.u.Interrupt.Level = 1;
3033 pCfg->KbdInt.u.Interrupt.Vector = 1;
3034 }
3035
3036 if (pCfg->cPorts == 0)
3037 {
3038 pCfg->aPorts[i8042Dat].Type = CmResourceTypePort;
3039 pCfg->aPorts[i8042Dat].Flags = CM_RESOURCE_PORT_IO;
3040 pCfg->aPorts[i8042Dat].ShareDisposition = CmResourceShareDriverExclusive;
3041 pCfg->aPorts[i8042Dat].u.Port.Start.LowPart = 0x60;
3042 pCfg->aPorts[i8042Dat].u.Port.Start.HighPart = 0;
3043 pCfg->aPorts[i8042Dat].u.Port.Length = 1;
3044
3045 pCfg->aPorts[i8042Cmd].Type = CmResourceTypePort;
3046 pCfg->aPorts[i8042Cmd].Flags = CM_RESOURCE_PORT_IO;
3047 pCfg->aPorts[i8042Cmd].ShareDisposition = CmResourceShareDriverExclusive;
3048 pCfg->aPorts[i8042Cmd].u.Port.Start.LowPart = 0x64;
3049 pCfg->aPorts[i8042Cmd].u.Port.Start.HighPart = 0;
3050 pCfg->aPorts[i8042Cmd].u.Port.Length = 1;
3051
3052 pCfg->cPorts = 2;
3053 }
3054 else if (pCfg->cPorts == 1)
3055 {
3056 pCfg->aPorts[i8042Dat].u.Port.Length = 1;
3057 pCfg->aPorts[i8042Cmd] = pCfg->aPorts[i8042Dat];
3058 pCfg->aPorts[i8042Cmd].u.Port.Start.LowPart += 4;
3059 pCfg->cPorts++;
3060 }
3061 else
3062 {
3063 if (pCfg->aPorts[i8042Cmd].u.Port.Start.LowPart < pCfg->aPorts[i8042Dat].u.Port.Start.LowPart)
3064 {
3065 CM_PARTIAL_RESOURCE_DESCRIPTOR Desc = pCfg->aPorts[i8042Dat];
3066 pCfg->aPorts[i8042Dat] = pCfg->aPorts[i8042Cmd];
3067 pCfg->aPorts[i8042Cmd] = Desc;
3068 }
3069 }
3070
3071 return STATUS_SUCCESS;
3072}
3073
3074/**
3075 * Callout routine. Grab the pointer controller and the peripheral
3076 * configuration information.
3077 */
3078static NTSTATUS MouCallOut(PVOID pCtx, PUNICODE_STRING PathName,
3079 INTERFACE_TYPE BusType, ULONG uBusNr, PKEY_VALUE_FULL_INFORMATION *pBusInf,
3080 CONFIGURATION_TYPE uCtrlType, ULONG uCtrlNr, PKEY_VALUE_FULL_INFORMATION *pCtrlInf,
3081 CONFIGURATION_TYPE uPrfType, ULONG uPrfNr, PKEY_VALUE_FULL_INFORMATION *pPrfInf)
3082{
3083 NTSTATUS status = STATUS_SUCCESS;
3084
3085 UNICODE_STRING unicodeIdentifier;
3086 GetDevIdentifier(pPrfInf, &unicodeIdentifier);
3087
3088 PINITEXT pInit = (PINITEXT)pCtx;
3089 PDEVEXT pDevExt = &pInit->DevExt;
3090
3091 if ( (pDevExt->HardwarePresent & MOUSE_HARDWARE_PRESENT)
3092 || unicodeIdentifier.Length == 0)
3093 return status;
3094
3095 ANSI_STRING ansiString;
3096 status = RtlUnicodeStringToAnsiString(&ansiString, &unicodeIdentifier, TRUE);
3097 if (!NT_SUCCESS(status))
3098 return status;
3099
3100 if (strstr(ansiString.Buffer, "PS2"))
3101 pDevExt->HardwarePresent |= MOUSE_HARDWARE_PRESENT;
3102
3103 RtlFreeAnsiString(&ansiString);
3104
3105 if (!(pDevExt->HardwarePresent & MOUSE_HARDWARE_PRESENT))
3106 return status;
3107
3108 PI8042CFGINF pCfg = &pDevExt->Cfg;
3109 if (!(pDevExt->HardwarePresent & KEYBOARD_HARDWARE_PRESENT))
3110 {
3111 pCfg->InterfaceType = BusType;
3112 pCfg->uBusNr = uBusNr;
3113 pCfg->fFloatSave = FALSE;
3114 }
3115
3116 BOOLEAN fDefIntShare;
3117 KINTERRUPT_MODE DefIntMode;
3118 if (pCfg->InterfaceType == MicroChannel)
3119 {
3120 fDefIntShare = TRUE;
3121 DefIntMode = LevelSensitive;
3122 }
3123 else
3124 {
3125 fDefIntShare = FALSE;
3126 DefIntMode = Latched;
3127 }
3128
3129 PCM_PARTIAL_RESOURCE_LIST pCtrlData;
3130 if (GetDevCfgData(pCtrlInf, &pCtrlData))
3131 {
3132 unsigned cList = pCtrlData->Count;
3133 PCM_PARTIAL_RESOURCE_DESCRIPTOR pResDesc = pCtrlData->PartialDescriptors;
3134 BOOLEAN fPortInfoNeeded = pCfg->cPorts ? FALSE : TRUE;
3135
3136 for (unsigned i = 0; i < cList; i++, pResDesc++)
3137 {
3138 switch (pResDesc->Type)
3139 {
3140 case CmResourceTypePort:
3141 if (fPortInfoNeeded)
3142 {
3143 pCfg->aPorts[pCfg->cPorts] = *pResDesc;
3144 pCfg->aPorts[pCfg->cPorts].ShareDisposition = CmResourceShareDriverExclusive;
3145 pCfg->cPorts++;
3146 }
3147 break;
3148
3149 case CmResourceTypeInterrupt:
3150 pCfg->MouInt = *pResDesc;
3151 pCfg->MouInt.ShareDisposition = fDefIntShare ? CmResourceShareShared
3152 : CmResourceShareDeviceExclusive;
3153 break;
3154
3155 default:
3156 break;
3157 }
3158 }
3159 }
3160
3161 if (!(pCfg->MouInt.Type & CmResourceTypeInterrupt))
3162 {
3163 pCfg->MouInt.Type = CmResourceTypeInterrupt;
3164 pCfg->MouInt.ShareDisposition = fDefIntShare ? CmResourceShareShared
3165 : CmResourceShareDeviceExclusive;
3166 pCfg->MouInt.Flags = (DefIntMode == Latched) ? CM_RESOURCE_INTERRUPT_LATCHED
3167 : CM_RESOURCE_INTERRUPT_LEVEL_SENSITIVE;
3168 pCfg->MouInt.u.Interrupt.Level = 12;
3169 pCfg->MouInt.u.Interrupt.Vector = 12;
3170 }
3171
3172 if (pCfg->cPorts == 0)
3173 {
3174 pCfg->aPorts[i8042Dat].Type = CmResourceTypePort;
3175 pCfg->aPorts[i8042Dat].Flags = CM_RESOURCE_PORT_IO;
3176 pCfg->aPorts[i8042Dat].ShareDisposition = CmResourceShareDriverExclusive;
3177 pCfg->aPorts[i8042Dat].u.Port.Start.LowPart = 0x60;
3178 pCfg->aPorts[i8042Dat].u.Port.Start.HighPart = 0;
3179 pCfg->aPorts[i8042Dat].u.Port.Length = 1;
3180
3181 pCfg->aPorts[i8042Cmd].Type = CmResourceTypePort;
3182 pCfg->aPorts[i8042Cmd].Flags = CM_RESOURCE_PORT_IO;
3183 pCfg->aPorts[i8042Cmd].ShareDisposition = CmResourceShareDriverExclusive;
3184 pCfg->aPorts[i8042Cmd].u.Port.Start.LowPart = 0x64;
3185 pCfg->aPorts[i8042Cmd].u.Port.Start.HighPart = 0;
3186 pCfg->aPorts[i8042Cmd].u.Port.Length = 1;
3187
3188 pCfg->cPorts = 2;
3189 }
3190 else if (pCfg->cPorts == 1)
3191 {
3192 pCfg->aPorts[i8042Cmd] = pCfg->aPorts[i8042Dat];
3193 pCfg->aPorts[i8042Cmd].u.Port.Start.LowPart += 4;
3194 pCfg->cPorts++;
3195 }
3196 else
3197 {
3198 if (pCfg->aPorts[i8042Cmd].u.Port.Start.LowPart < pCfg->aPorts[i8042Dat].u.Port.Start.LowPart)
3199 {
3200 CM_PARTIAL_RESOURCE_DESCRIPTOR Desc = pCfg->aPorts[i8042Dat];
3201 pCfg->aPorts[i8042Dat] = pCfg->aPorts[i8042Cmd];
3202 pCfg->aPorts[i8042Cmd] = Desc;
3203 }
3204 }
3205
3206 return status;
3207}
3208
3209static const UCHAR s_ucCommands[] =
3210{
3211 SET_MOUSE_SAMPLING_RATE, 200,
3212 SET_MOUSE_SAMPLING_RATE, 100,
3213 SET_MOUSE_SAMPLING_RATE, 80,
3214 GET_DEVICE_ID, 0
3215};
3216
3217static NTSTATUS MouFindWheel(PDEVICE_OBJECT pDevObj)
3218{
3219 NTSTATUS status;
3220 PDEVEXT pDevExt = (PDEVEXT) pDevObj->DeviceExtension;
3221
3222 if (!pDevExt->Cfg.EnableWheelDetection)
3223 return STATUS_NO_SUCH_DEVICE;
3224
3225 KeStallExecutionProcessor(50);
3226
3227 for (unsigned iCmd = 0; s_ucCommands[iCmd];)
3228 {
3229 status = PutBytePoll(i8042Dat, TRUE /*=wait*/, MouDevType, pDevExt, s_ucCommands[iCmd]);
3230 if (!NT_SUCCESS(status))
3231 goto fail;
3232
3233 iCmd++;
3234 KeStallExecutionProcessor(50);
3235 }
3236
3237 UCHAR byte;
3238 for (unsigned i = 0; i < 5; i++)
3239 {
3240 status = GetBytePoll(CtrlDevType, pDevExt, &byte);
3241 if (status != STATUS_IO_TIMEOUT)
3242 break;
3243 KeStallExecutionProcessor(50);
3244 }
3245
3246 if ( NT_SUCCESS(status)
3247 && (byte == MOUSE_ID_BYTE || byte == WHEELMOUSE_ID_BYTE))
3248 {
3249 if (byte == WHEELMOUSE_ID_BYTE)
3250 {
3251 pDevExt->HardwarePresent |= (WHEELMOUSE_HARDWARE_PRESENT | MOUSE_HARDWARE_PRESENT);
3252 pDevExt->Cfg.MouAttr.MouseIdentifier = WHEELMOUSE_I8042_HARDWARE;
3253 }
3254 else
3255 pDevExt->HardwarePresent |= MOUSE_HARDWARE_PRESENT;
3256 }
3257
3258fail:
3259 pDevExt->MouExt.uPrevSignAndOverflow = 0;
3260 pDevExt->MouExt.InputState = MouseExpectingACK;
3261 return status;
3262}
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