Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1025 lines
30 KiB

  1. /*++
  2. Copyright (c) 1996 Microsoft Corporation
  3. Module Name:
  4. hid.c
  5. Abstract:
  6. This module contains the code for translating HID reports to keyboard
  7. reports.
  8. Environment:
  9. Kernel & user mode
  10. Revision History:
  11. Nov-96 : Created by Kenneth D. Ray
  12. --*/
  13. //
  14. // For this module only we set the INITGUID macro before including wdm and
  15. // hidclass.h This not only declares the guids but also initializes them.
  16. //
  17. #include "kbdhid.h"
  18. #include "hidclass.h"
  19. #include <initguid.h>
  20. #include <wdmguid.h>
  21. #ifdef ALLOC_PRAGMA
  22. #pragma alloc_text(PAGE,KbdHid_CallHidClass)
  23. #pragma alloc_text(PAGE,KbdHid_AddDevice)
  24. #pragma alloc_text(PAGE,KbdHid_StartDevice)
  25. #pragma alloc_text(PAGE,KbdHid_PnP)
  26. #endif
  27. NTSTATUS
  28. KbdHid_CallHidClass(
  29. IN PDEVICE_EXTENSION Data,
  30. IN ULONG Ioctl,
  31. PVOID InputBuffer,
  32. ULONG InputBufferLength,
  33. PVOID OutputBuffer,
  34. ULONG OutputBufferLength
  35. )
  36. /*++
  37. Routine Description:
  38. Make a *synchronous* request of the HID class driver
  39. Arguments:
  40. Ioctl - Value of the IOCTL request.
  41. InputBuffer - Buffer to be sent to the HID class driver.
  42. InputBufferLength - Size of buffer to be sent to the HID class driver.
  43. OutputBuffer - Buffer for received data from the HID class driver.
  44. OutputBufferLength - Size of receive buffer from the HID class.
  45. Return Value:
  46. STATUS_SUCCESS if successful,
  47. STATUS_UNSUCCESSFUL otherwise
  48. --*/
  49. {
  50. KEVENT event;
  51. PIRP irp;
  52. IO_STATUS_BLOCK ioStatus;
  53. PIO_STACK_LOCATION nextStack;
  54. NTSTATUS status = STATUS_SUCCESS;
  55. PAGED_CODE ();
  56. Print(DBG_PNP_TRACE, ("PNP-CallHidClass: Enter." ));
  57. //
  58. // issue a synchronous request
  59. //
  60. KeInitializeEvent(&event, NotificationEvent, FALSE);
  61. irp = IoBuildDeviceIoControlRequest (
  62. Ioctl,
  63. Data->TopOfStack,
  64. InputBuffer,
  65. InputBufferLength,
  66. OutputBuffer,
  67. OutputBufferLength,
  68. FALSE, // External
  69. &event,
  70. &ioStatus);
  71. if (NULL == irp) {
  72. return STATUS_UNSUCCESSFUL;
  73. }
  74. //
  75. // Call the class driver to perform the operation. If the returned status
  76. // is PENDING, wait for the request to complete.
  77. //
  78. nextStack = IoGetNextIrpStackLocation(irp);
  79. ASSERT(nextStack != NULL);
  80. status = IoCallDriver(Data->TopOfStack, irp);
  81. if (status == STATUS_PENDING) {
  82. status = KeWaitForSingleObject(
  83. &event,
  84. Executive, // wait reason
  85. KernelMode,
  86. FALSE, // we are not alertable
  87. NULL); // No time out !!!!
  88. }
  89. if (NT_SUCCESS (status)) {
  90. status = ioStatus.Status;
  91. }
  92. Print(DBG_PNP_TRACE, ("PNP-CallHidClass: Enter." ));
  93. return status;
  94. }
  95. NTSTATUS
  96. KbdHid_QueryDeviceKey (
  97. IN HANDLE Handle,
  98. IN PWCHAR ValueNameString,
  99. OUT PVOID Data,
  100. IN ULONG DataLength
  101. )
  102. {
  103. NTSTATUS status;
  104. UNICODE_STRING valueName;
  105. ULONG length;
  106. PKEY_VALUE_FULL_INFORMATION fullInfo;
  107. RtlInitUnicodeString (&valueName, ValueNameString);
  108. length = sizeof (KEY_VALUE_FULL_INFORMATION)
  109. + valueName.MaximumLength
  110. + DataLength;
  111. fullInfo = ExAllocatePool (PagedPool, length);
  112. if (fullInfo) {
  113. status = ZwQueryValueKey (Handle,
  114. &valueName,
  115. KeyValueFullInformation,
  116. fullInfo,
  117. length,
  118. &length);
  119. if (NT_SUCCESS (status)) {
  120. ASSERT (DataLength == fullInfo->DataLength);
  121. RtlCopyMemory (Data,
  122. ((PUCHAR) fullInfo) + fullInfo->DataOffset,
  123. fullInfo->DataLength);
  124. }
  125. ExFreePool (fullInfo);
  126. } else {
  127. status = STATUS_NO_MEMORY;
  128. }
  129. return status;
  130. }
  131. NTSTATUS
  132. KbdHid_AddDevice (
  133. IN PDRIVER_OBJECT Driver,
  134. IN PDEVICE_OBJECT PDO
  135. )
  136. /*++
  137. Routine Description:
  138. Arguments:
  139. Return Value:
  140. NTSTATUS result code.
  141. --*/
  142. {
  143. NTSTATUS status = STATUS_SUCCESS;
  144. PDEVICE_EXTENSION data;
  145. PDEVICE_OBJECT device;
  146. POWER_STATE state;
  147. PAGED_CODE ();
  148. Print (DBG_PNP_TRACE, ("enter Add Device \n"));
  149. status = IoCreateDevice(Driver,
  150. sizeof(DEVICE_EXTENSION),
  151. NULL, // no name for this Filter DO
  152. FILE_DEVICE_KEYBOARD,
  153. 0,
  154. FALSE,
  155. &device);
  156. if (!NT_SUCCESS (status)) {
  157. return(status);
  158. }
  159. data = (PDEVICE_EXTENSION) device->DeviceExtension;
  160. //
  161. // Initialize the fields.
  162. //
  163. data->TopOfStack = IoAttachDeviceToDeviceStack (device, PDO);
  164. if (data->TopOfStack == NULL) {
  165. PIO_ERROR_LOG_PACKET errorLogEntry;
  166. //
  167. // Not good; in only extreme cases will this fail
  168. //
  169. errorLogEntry = (PIO_ERROR_LOG_PACKET)
  170. IoAllocateErrorLogEntry(Driver,
  171. (UCHAR) sizeof(IO_ERROR_LOG_PACKET));
  172. if (errorLogEntry) {
  173. errorLogEntry->ErrorCode = KBDHID_ATTACH_DEVICE_FAILED;
  174. errorLogEntry->DumpDataSize = 0;
  175. errorLogEntry->SequenceNumber = 0;
  176. errorLogEntry->MajorFunctionCode = 0;
  177. errorLogEntry->IoControlCode = 0;
  178. errorLogEntry->RetryCount = 0;
  179. errorLogEntry->UniqueErrorValue = 0;
  180. errorLogEntry->FinalStatus = STATUS_DEVICE_NOT_CONNECTED;
  181. IoWriteErrorLogEntry(errorLogEntry);
  182. }
  183. IoDeleteDevice(device);
  184. return STATUS_DEVICE_NOT_CONNECTED;
  185. }
  186. ASSERT (data->TopOfStack);
  187. data->Self = device;
  188. data->Started = FALSE;
  189. data->Initialized = FALSE;
  190. data->UnitId = (USHORT) InterlockedIncrement (&Globals.UnitId);
  191. data->PDO = PDO;
  192. KeInitializeSpinLock(&data->usageMappingSpinLock);
  193. data->ReadIrp = IoAllocateIrp (data->TopOfStack->StackSize, FALSE);
  194. // Initializiation happens automatically.
  195. if (NULL == data->ReadIrp) {
  196. IoDetachDevice (data->TopOfStack);
  197. IoDeleteDevice (device);
  198. return STATUS_INSUFFICIENT_RESOURCES;
  199. }
  200. KeInitializeEvent (&data->ReadCompleteEvent, SynchronizationEvent, FALSE);
  201. KeInitializeEvent (&data->ReadSentEvent, NotificationEvent, TRUE);
  202. IoInitializeRemoveLock (&data->RemoveLock, KBDHID_POOL_TAG, 1, 10);
  203. data->ReadFile = NULL;
  204. ExInitializeFastMutex (&data->CreateCloseMutex);
  205. data->InputData.UnitId = data->UnitId;
  206. data->InputData.MakeCode = 0;
  207. data->InputData.Flags = 0;
  208. data->ScanState = Normal;
  209. //
  210. // Initialize the keyboard attributes structure. This information is
  211. // queried via IOCTL_KEYBOARD_QUERY_ATTRIBUTES. [DAN]
  212. //
  213. data->Attributes.KeyboardIdentifier.Type = HID_KEYBOARD_IDENTIFIER_TYPE;
  214. data->Attributes.KeyboardIdentifier.Subtype = 0;
  215. data->IdEx.Type = HID_KEYBOARD_IDENTIFIER_TYPE;
  216. data->IdEx.Subtype = 0;
  217. data->Attributes.KeyboardMode = HID_KEYBOARD_SCAN_CODE_SET;
  218. data->Attributes.NumberOfFunctionKeys = HID_KEYBOARD_NUMBER_OF_FUNCTION_KEYS;
  219. data->Attributes.NumberOfIndicators = HID_KEYBOARD_NUMBER_OF_INDICATORS;
  220. data->Attributes.NumberOfKeysTotal = HID_KEYBOARD_NUMBER_OF_KEYS_TOTAL;
  221. data->Attributes.InputDataQueueLength = 1;
  222. data->Attributes.KeyRepeatMinimum.UnitId = data->UnitId;
  223. data->Attributes.KeyRepeatMinimum.Rate = HID_KEYBOARD_TYPEMATIC_RATE_MINIMUM;
  224. data->Attributes.KeyRepeatMinimum.Delay = HID_KEYBOARD_TYPEMATIC_DELAY_MINIMUM;
  225. data->Attributes.KeyRepeatMaximum.UnitId = data->UnitId;
  226. data->Attributes.KeyRepeatMaximum.Rate = HID_KEYBOARD_TYPEMATIC_RATE_MAXIMUM;
  227. data->Attributes.KeyRepeatMaximum.Delay = HID_KEYBOARD_TYPEMATIC_DELAY_MAXIMUM;
  228. //
  229. // Initialize the keyboard indicators structure. [DAN]
  230. //
  231. data->Indicators.UnitId = data->UnitId;
  232. data->Indicators.LedFlags = 0;
  233. //
  234. // Initialize the keyboard typematic info structure. [DAN]
  235. //
  236. data->Typematic.UnitId = data->UnitId;
  237. data->Typematic.Rate = HID_KEYBOARD_TYPEMATIC_RATE_DEFAULT;
  238. data->Typematic.Delay = HID_KEYBOARD_TYPEMATIC_DELAY_DEFAULT;
  239. //
  240. // Initialize private typematic information. [DAN]
  241. //
  242. KeInitializeDpc (&data->AutoRepeatDPC, KbdHid_AutoRepeat, data);
  243. KeInitializeTimer (&data->AutoRepeatTimer);
  244. data->AutoRepeatRate = 1000 / HID_KEYBOARD_TYPEMATIC_RATE_DEFAULT; //ms
  245. data->AutoRepeatDelay.LowPart = -HID_KEYBOARD_TYPEMATIC_DELAY_DEFAULT * 10000;
  246. //100ns
  247. data->AutoRepeatDelay.HighPart = -1;
  248. #if KEYBOARD_HW_CHATTERY_FIX // [DAN]
  249. //
  250. // Initialize StartRead-initiator DPC.
  251. //
  252. KeInitializeDpc (&data->InitiateStartReadDPC,
  253. KbdHid_InitiateStartRead,
  254. data);
  255. KeInitializeTimer (&data->InitiateStartReadTimer);
  256. data->InitiateStartReadDelay.QuadPart = -DEFAULT_START_READ_DELAY;
  257. data->InitiateStartReadUserNotified = FALSE;
  258. #endif
  259. state.DeviceState = PowerDeviceD0;
  260. PoSetPowerState (device, DevicePowerState, state);
  261. data->WmiLibInfo.GuidCount = sizeof (KbdHid_WmiGuidList) /
  262. sizeof (WMIGUIDREGINFO);
  263. data->WmiLibInfo.GuidList = KbdHid_WmiGuidList;
  264. data->WmiLibInfo.QueryWmiRegInfo = KbdHid_QueryWmiRegInfo;
  265. data->WmiLibInfo.QueryWmiDataBlock = KbdHid_QueryWmiDataBlock;
  266. data->WmiLibInfo.SetWmiDataBlock = KbdHid_SetWmiDataBlock;
  267. data->WmiLibInfo.SetWmiDataItem = KbdHid_SetWmiDataItem;
  268. data->WmiLibInfo.ExecuteWmiMethod = NULL;
  269. data->WmiLibInfo.WmiFunctionControl = NULL;
  270. device->Flags |= DO_POWER_PAGABLE;
  271. device->Flags &= ~DO_DEVICE_INITIALIZING;
  272. return status;
  273. }
  274. NTSTATUS
  275. KbdHid_StartDevice (
  276. IN PDEVICE_EXTENSION Data
  277. )
  278. /*++
  279. Routine Description:
  280. Arguments:
  281. Return Value:
  282. NTSTATUS result code.
  283. --*/
  284. {
  285. HIDP_CAPS caps; // the capabilities of the found hid device
  286. HID_COLLECTION_INFORMATION info;
  287. NTSTATUS status = STATUS_SUCCESS;
  288. PHIDP_PREPARSED_DATA preparsedData = NULL;
  289. PHID_EXTENSION hid = NULL;
  290. ULONG length, usageListLength, inputBufferLength;
  291. ULONG maxUsages;
  292. PCHAR buffer;
  293. HANDLE devInstRegKey;
  294. ULONG tmp;
  295. PAGED_CODE ();
  296. Print (DBG_PNP_TRACE, ("enter START Device \n"));
  297. //
  298. // Check the registry for any usage mapping information
  299. // for this particular keyboard.
  300. //
  301. // Note: Need to call this after devnode created
  302. // (after START_DEVICE completes).
  303. // For raw devices, this will fail on the first start
  304. // (b/c devnode not there yet)
  305. // but succeed on the second start.
  306. //
  307. LoadKeyboardUsageMappingList (Data);
  308. //
  309. // Retrieve the capabilities of this hid device
  310. // IOCTL_HID_GET_COLLECTION_INFORMATION fills in HID_COLLECTION_INFORMATION.
  311. // we are interested in the Descriptor Size, which tells us how big a
  312. // buffer to allocate for the preparsed data.
  313. //
  314. if (!NT_SUCCESS (status = KbdHid_CallHidClass (
  315. Data,
  316. IOCTL_HID_GET_COLLECTION_INFORMATION,
  317. 0, 0, // no input
  318. &info, sizeof (info)))) {
  319. goto KbdHid_StartDeviceReject;
  320. }
  321. //
  322. // Allocate memory to hold the preparsed data.
  323. //
  324. preparsedData = (PHIDP_PREPARSED_DATA)
  325. ExAllocatePool (NonPagedPool, info.DescriptorSize);
  326. if (!preparsedData) {
  327. status = STATUS_INSUFFICIENT_RESOURCES;
  328. goto KbdHid_StartDeviceReject;
  329. }
  330. //
  331. // Retrieve that information.
  332. //
  333. if (!NT_SUCCESS (status = KbdHid_CallHidClass (
  334. Data,
  335. IOCTL_HID_GET_COLLECTION_DESCRIPTOR,
  336. 0, 0, // no input
  337. preparsedData, info.DescriptorSize))) {
  338. goto KbdHid_StartDeviceReject;
  339. }
  340. //
  341. // Call the parser to determine the capabilites of this HID device.
  342. //
  343. if (!NT_SUCCESS (status = HidP_GetCaps (preparsedData, &caps))) {
  344. goto KbdHid_StartDeviceReject;
  345. }
  346. #if 0
  347. //
  348. // Is this the thing we want?
  349. //
  350. // In this particular case we are looking for a keyboard.
  351. //
  352. if ((HID_USAGE_PAGE_GENERIC == caps.UsagePage) &&
  353. (HID_USAGE_GENERIC_KEYBOARD == caps.Usage)) {
  354. ;
  355. } else {
  356. //
  357. // Someone made an INF blunder!
  358. //
  359. ASSERT ((HID_USAGE_PAGE_GENERIC == caps.UsagePage) &&
  360. (HID_USAGE_GENERIC_KEYBOARD == caps.Usage));
  361. status = STATUS_UNSUCCESSFUL;
  362. goto KbdHid_StartDeviceReject;
  363. }
  364. #endif
  365. //
  366. // Set the number of buttons for this keyboard.
  367. // Note: we are actually reading here the total number of independant
  368. // chanels on the device. But that should be satisfactory for a keyboard.
  369. //
  370. Data->Attributes.NumberOfKeysTotal = caps.NumberInputDataIndices;
  371. //
  372. // look for any device parameters.
  373. //
  374. status = IoOpenDeviceRegistryKey (Data->PDO,
  375. PLUGPLAY_REGKEY_DEVICE,
  376. STANDARD_RIGHTS_ALL,
  377. &devInstRegKey);
  378. if (NT_SUCCESS (status)) {
  379. status = KbdHid_QueryDeviceKey (devInstRegKey,
  380. KEYBOARD_TYPE_OVERRIDE,
  381. &tmp,
  382. sizeof (tmp));
  383. if (NT_SUCCESS (status)) {
  384. Data->Attributes.KeyboardIdentifier.Type = (UCHAR) tmp;
  385. Data->IdEx.Type = tmp;
  386. }
  387. status = KbdHid_QueryDeviceKey (devInstRegKey,
  388. KEYBOARD_SUBTYPE_OVERRIDE,
  389. &tmp,
  390. sizeof (tmp));
  391. if (NT_SUCCESS (status)) {
  392. Data->Attributes.KeyboardIdentifier.Subtype = (UCHAR) tmp;
  393. Data->IdEx.Subtype = tmp;
  394. }
  395. status = KbdHid_QueryDeviceKey (devInstRegKey,
  396. KEYBOARD_NUMBER_TOTAL_KEYS_OVERRIDE,
  397. &tmp,
  398. sizeof (tmp));
  399. if (NT_SUCCESS (status)) {
  400. Data->Attributes.NumberOfKeysTotal = (USHORT) tmp;
  401. }
  402. status = KbdHid_QueryDeviceKey (devInstRegKey,
  403. KEYBOARD_NUMBER_FUNCTION_KEYS_OVERRIDE,
  404. &tmp,
  405. sizeof (tmp));
  406. if (NT_SUCCESS (status)) {
  407. Data->Attributes.NumberOfFunctionKeys = (USHORT) tmp;
  408. }
  409. status = KbdHid_QueryDeviceKey (devInstRegKey,
  410. KEYBOARD_NUMBER_INDICATORS_OVERRIDE,
  411. &tmp,
  412. sizeof (tmp));
  413. if (NT_SUCCESS (status)) {
  414. Data->Attributes.NumberOfIndicators = (USHORT) tmp;
  415. }
  416. ZwClose (devInstRegKey);
  417. if (!NT_SUCCESS (status)) {
  418. status = STATUS_SUCCESS;
  419. }
  420. }
  421. //
  422. // Note: here we might also want to check the button and value capabilities
  423. // of the device as well.
  424. //
  425. // Then let's use it.
  426. //
  427. //
  428. // a buffer length to allow an Input buffer, output buffer, feature buffer,
  429. // and the total number of usages that can be returned from a read packet.
  430. //
  431. maxUsages = HidP_MaxUsageListLength (
  432. HidP_Input,
  433. HID_USAGE_PAGE_KEYBOARD,
  434. preparsedData);
  435. //
  436. // Create space in the device extension for the buffer storage when working
  437. // with this HID device.
  438. //
  439. // We need four buffers to hold the button codes (length returned from
  440. // HidP_MaxUsageListLength) this will hold the current list of usages,
  441. // the previous list of usages, the ``Make'' and the ``Break'' lists.
  442. // We also need a place to put the input, output, and feature report
  443. // buffers.
  444. //
  445. usageListLength = ALIGNPTRLEN(maxUsages * sizeof (USAGE_AND_PAGE));
  446. inputBufferLength = ALIGNPTRLEN(caps.InputReportByteLength);
  447. length = (6 * usageListLength)
  448. + inputBufferLength
  449. + sizeof (HID_EXTENSION);
  450. Data->HidExtension = hid = ExAllocatePool (NonPagedPool, length);
  451. if (!hid) {
  452. status = STATUS_INSUFFICIENT_RESOURCES;
  453. goto KbdHid_StartDeviceReject;
  454. }
  455. RtlZeroMemory (hid, length);
  456. //
  457. // Initialize the fields.
  458. //
  459. hid->Ppd = preparsedData;
  460. hid->Caps = caps;
  461. hid->MaxUsages = maxUsages;
  462. // hid->ModifierState.ul = 0;
  463. hid->InputBuffer = buffer = hid->Buffer;
  464. hid->PreviousUsageList = (PUSAGE_AND_PAGE) (buffer += inputBufferLength);
  465. hid->CurrentUsageList = (PUSAGE_AND_PAGE) (buffer += usageListLength);
  466. hid->BreakUsageList = (PUSAGE_AND_PAGE) (buffer += usageListLength);
  467. hid->MakeUsageList = (PUSAGE_AND_PAGE) (buffer += usageListLength);
  468. hid->OldMakeUsageList = (PUSAGE_AND_PAGE) (buffer += usageListLength);
  469. hid->ScrapBreakUsageList = (PUSAGE_AND_PAGE) (buffer + usageListLength);
  470. //
  471. // Create the MDLs
  472. // HidClass uses direct IO so you need MDLs
  473. //
  474. hid->InputMdl = IoAllocateMdl (hid->InputBuffer, // The virtual address
  475. caps.InputReportByteLength, // length
  476. FALSE, // No associated IRP => not secondary
  477. FALSE, // No quota charge
  478. 0); // No associated IRP
  479. if (NULL == hid->InputMdl) {
  480. status = STATUS_INSUFFICIENT_RESOURCES;
  481. goto KbdHid_StartDeviceReject;
  482. }
  483. MmBuildMdlForNonPagedPool (hid->InputMdl); // Build this MDL.
  484. return status;
  485. KbdHid_StartDeviceReject:
  486. if (preparsedData) {
  487. // no need to set hid->Ppd to NULL becuase we will be freeing it as well
  488. ExFreePool (preparsedData);
  489. }
  490. if (hid) {
  491. if (hid->InputMdl) {
  492. IoFreeMdl (hid->InputMdl);
  493. }
  494. ExFreePool (hid);
  495. Data->HidExtension = NULL;
  496. }
  497. return status;
  498. }
  499. NTSTATUS
  500. KbdHid_PnP (
  501. IN PDEVICE_OBJECT DeviceObject,
  502. IN PIRP Irp
  503. )
  504. /*++
  505. Routine Description:
  506. The plug and play dispatch routines.
  507. Most of these this filter driver will completely ignore.
  508. In all cases it must pass on the IRP to the lower driver.
  509. Arguments:
  510. DeviceObject - pointer to a device object.
  511. Irp - pointer to an I/O Request Packet.
  512. Return Value:
  513. NT status code
  514. --*/
  515. {
  516. PDEVICE_EXTENSION data;
  517. PHID_EXTENSION hid;
  518. PIO_STACK_LOCATION stack;
  519. NTSTATUS status;
  520. ULONG i, j;
  521. PDEVICE_EXTENSION * classDataList;
  522. LARGE_INTEGER time;
  523. PAGED_CODE ();
  524. data = (PDEVICE_EXTENSION) DeviceObject->DeviceExtension;
  525. stack = IoGetCurrentIrpStackLocation (Irp);
  526. hid = data->HidExtension;
  527. status = IoAcquireRemoveLock (&data->RemoveLock, Irp);
  528. if (!NT_SUCCESS (status)) {
  529. //
  530. // Someone gave us a pnp irp after a remove. Unthinkable!
  531. //
  532. ASSERT (FALSE);
  533. Irp->IoStatus.Information = 0;
  534. Irp->IoStatus.Status = status;
  535. IoCompleteRequest (Irp, IO_NO_INCREMENT);
  536. return status;
  537. }
  538. Print(DBG_PNP_TRACE, ("PNP: Minor code = %x.", stack->MinorFunction));
  539. switch (stack->MinorFunction) {
  540. case IRP_MN_START_DEVICE:
  541. if (data->Started) {
  542. Print(DBG_PNP_INFO, ("PNP: Device already started." ));
  543. status = STATUS_SUCCESS;
  544. Irp->IoStatus.Status = status;
  545. IoCompleteRequest (Irp, IO_NO_INCREMENT);
  546. break;
  547. }
  548. //
  549. // The device is starting.
  550. //
  551. // We cannot touch the device (send it any non pnp irps) until a
  552. // start device has been passed down to the lower drivers.
  553. //
  554. IoCopyCurrentIrpStackLocationToNext (Irp);
  555. KeInitializeEvent(&data->StartEvent, NotificationEvent, FALSE);
  556. IoSetCompletionRoutine (Irp,
  557. KbdHid_PnPComplete,
  558. data,
  559. TRUE,
  560. TRUE,
  561. TRUE); // No need for Cancel
  562. Irp->IoStatus.Status = STATUS_SUCCESS;
  563. status = IoCallDriver (data->TopOfStack, Irp);
  564. if (STATUS_PENDING == status) {
  565. KeWaitForSingleObject(
  566. &data->StartEvent,
  567. Executive, // Waiting for reason of a driver
  568. KernelMode, // Waiting in kernel mode
  569. FALSE, // No allert
  570. NULL); // No timeout
  571. }
  572. if (NT_SUCCESS (status) && NT_SUCCESS (Irp->IoStatus.Status)) {
  573. //
  574. // As we are successfully now back from our start device
  575. // we can do work.
  576. //
  577. if (!data->Initialized) {
  578. status = KbdHid_StartDevice (data);
  579. if (NT_SUCCESS (status)) {
  580. IoWMIRegistrationControl(DeviceObject,
  581. WMIREG_ACTION_REGISTER
  582. );
  583. data->Started = TRUE;
  584. data->Initialized = TRUE;
  585. }
  586. } else {
  587. data->Started = TRUE;
  588. }
  589. }
  590. //
  591. // We must now complete the IRP, since we stopped it in the
  592. // completetion routine with MORE_PROCESSING_REQUIRED.
  593. //
  594. Irp->IoStatus.Status = status;
  595. Irp->IoStatus.Information = 0;
  596. IoCompleteRequest (Irp, IO_NO_INCREMENT);
  597. break;
  598. case IRP_MN_STOP_DEVICE:
  599. //
  600. // After the start IRP has been sent to the lower driver object, the
  601. // bus may NOT send any more IRPS down ``touch'' until another START
  602. // has occured.
  603. // What ever access is required must be done before the Irp is passed
  604. // on.
  605. //
  606. if (data->Started) {
  607. //
  608. // Do what ever
  609. //
  610. }
  611. //
  612. // We don't need a completion routine so fire and forget.
  613. //
  614. // Set the current stack location to the next stack location and
  615. // call the next device object.
  616. //
  617. //
  618. // Stop Device touching the hardware MouStopDevice(data, TRUE);
  619. //
  620. data->Started = FALSE;
  621. Irp->IoStatus.Status = STATUS_SUCCESS;
  622. IoSkipCurrentIrpStackLocation (Irp);
  623. status = IoCallDriver (data->TopOfStack, Irp);
  624. break;
  625. case IRP_MN_REMOVE_DEVICE:
  626. //
  627. // The PlugPlay system has detected the removal of this device. We
  628. // have no choise but to detach and delete the device objecct.
  629. // (If we wanted to express and interest in preventing this removal,
  630. // we should have filtered the query remove and query stop routines.)
  631. //
  632. // Note! we might receive a remove WITHOUT first receiving a stop.
  633. // ASSERT (!usbData->Removed);
  634. Print (DBG_PNP_TRACE, ("enter RemoveDevice \n"));
  635. IoWMIRegistrationControl(data->Self,
  636. WMIREG_ACTION_DEREGISTER
  637. );
  638. if (data->Started) {
  639. // Stop the device without touching the hardware.
  640. // MouStopDevice(data, FALSE);
  641. }
  642. //
  643. // Here if we had any outstanding requests in a personal queue we should
  644. // complete them all now.
  645. //
  646. // Note, the device could be GONE so we cannot send it any non-
  647. // PNP IRPS.
  648. //
  649. time = data->AutoRepeatDelay;
  650. KeCancelTimer (&data->AutoRepeatTimer);
  651. #if KEYBOARD_HW_CHATTERY_FIX
  652. KeCancelTimer (&data->InitiateStartReadTimer);
  653. //
  654. // NB the time is a negative (relative) number;
  655. //
  656. if (data->InitiateStartReadDelay.QuadPart < time.QuadPart) {
  657. time = data->InitiateStartReadDelay;
  658. }
  659. #endif
  660. KeDelayExecutionThread (KernelMode, FALSE, &time);
  661. //
  662. // Cancel our read IRP. [DAN]
  663. // Note - waiting is only really necessary on 98, where pnp doesn't
  664. // make sure all handles are closed before sending the remove.
  665. //
  666. data->ShuttingDown = TRUE;
  667. KeWaitForSingleObject (&data->ReadSentEvent,
  668. Executive,
  669. KernelMode,
  670. FALSE,
  671. NULL
  672. );
  673. IoCancelIrp (data->ReadIrp);
  674. //
  675. // Send on the remove IRP
  676. //
  677. Irp->IoStatus.Status = STATUS_SUCCESS;
  678. IoSkipCurrentIrpStackLocation (Irp);
  679. status = IoCallDriver (data->TopOfStack, Irp);
  680. //
  681. // Wait for the remove lock to free.
  682. //
  683. IoReleaseRemoveLockAndWait (&data->RemoveLock, Irp);
  684. //
  685. // Free the associated memory.
  686. //
  687. IoFreeIrp (data->ReadIrp);
  688. if (hid) {
  689. //
  690. // If we are removed without being started then we will have
  691. // no hid extension
  692. //
  693. ExFreePool (hid->Ppd);
  694. IoFreeMdl (hid->InputMdl);
  695. ExFreePool (hid);
  696. }
  697. FreeKeyboardUsageMappingList(data);
  698. IoDetachDevice (data->TopOfStack);
  699. IoDeleteDevice (data->Self);
  700. return status;
  701. case IRP_MN_SURPRISE_REMOVAL:
  702. case IRP_MN_QUERY_REMOVE_DEVICE:
  703. case IRP_MN_CANCEL_REMOVE_DEVICE:
  704. case IRP_MN_QUERY_STOP_DEVICE:
  705. case IRP_MN_CANCEL_STOP_DEVICE:
  706. //
  707. // These IRPs have to have their status changed from
  708. // STATUS_NOT_SUPPORTED b4 passing them down.
  709. //
  710. Irp->IoStatus.Status = STATUS_SUCCESS;
  711. case IRP_MN_QUERY_DEVICE_RELATIONS:
  712. case IRP_MN_QUERY_INTERFACE:
  713. case IRP_MN_QUERY_CAPABILITIES:
  714. case IRP_MN_QUERY_RESOURCES:
  715. case IRP_MN_QUERY_RESOURCE_REQUIREMENTS:
  716. case IRP_MN_READ_CONFIG:
  717. case IRP_MN_WRITE_CONFIG:
  718. case IRP_MN_EJECT:
  719. case IRP_MN_SET_LOCK:
  720. case IRP_MN_QUERY_ID:
  721. case IRP_MN_QUERY_PNP_DEVICE_STATE:
  722. default:
  723. //
  724. // Here the filter driver might modify the behavior of these IRPS
  725. // Please see PlugPlay documentation for use of these IRPs.
  726. //
  727. IoSkipCurrentIrpStackLocation (Irp);
  728. status = IoCallDriver (data->TopOfStack, Irp);
  729. break;
  730. }
  731. IoReleaseRemoveLock (&data->RemoveLock, Irp);
  732. return status;
  733. }
  734. NTSTATUS
  735. KbdHid_PnPComplete (
  736. IN PDEVICE_OBJECT DeviceObject,
  737. IN PIRP Irp,
  738. IN PVOID Context
  739. )
  740. /*++
  741. Routine Description:
  742. The pnp IRP is in the process of completing.
  743. signal
  744. Arguments:
  745. Context set to the device object in question.
  746. --*/
  747. {
  748. PIO_STACK_LOCATION stack;
  749. PDEVICE_EXTENSION data;
  750. NTSTATUS status;
  751. UNREFERENCED_PARAMETER (DeviceObject);
  752. status = STATUS_SUCCESS;
  753. data = (PDEVICE_EXTENSION) Context;
  754. stack = IoGetCurrentIrpStackLocation (Irp);
  755. if (Irp->PendingReturned) {
  756. IoMarkIrpPending( Irp );
  757. }
  758. switch (stack->MajorFunction) {
  759. case IRP_MJ_PNP:
  760. switch (stack->MinorFunction) {
  761. case IRP_MN_START_DEVICE:
  762. KeSetEvent (&data->StartEvent, 0, FALSE);
  763. //
  764. // Take the IRP back so that we can continue using it during
  765. // the IRP_MN_START_DEVICE dispatch routine.
  766. // NB: we will have to call IoCompleteRequest
  767. //
  768. return STATUS_MORE_PROCESSING_REQUIRED;
  769. default:
  770. break;
  771. }
  772. break;
  773. case IRP_MJ_POWER:
  774. default:
  775. break;
  776. }
  777. return status;
  778. }
  779. NTSTATUS
  780. KbdHid_Power (
  781. IN PDEVICE_OBJECT DeviceObject,
  782. IN PIRP Irp
  783. )
  784. {
  785. PIO_STACK_LOCATION stack;
  786. NTSTATUS status;
  787. PDEVICE_EXTENSION data;
  788. POWER_STATE powerState;
  789. POWER_STATE_TYPE powerType;
  790. Print(DBG_POWER_TRACE, ("Power Enter." ));
  791. data = (PDEVICE_EXTENSION) DeviceObject->DeviceExtension;
  792. stack = IoGetCurrentIrpStackLocation (Irp);
  793. powerType = stack->Parameters.Power.Type;
  794. powerState = stack->Parameters.Power.State;
  795. status = IoAcquireRemoveLock (&data->RemoveLock, Irp);
  796. if (!NT_SUCCESS (status)) {
  797. PoStartNextPowerIrp (Irp);
  798. Irp->IoStatus.Status = status;
  799. IoCompleteRequest (Irp, IO_NO_INCREMENT);
  800. return status;
  801. }
  802. switch (stack->MinorFunction) {
  803. case IRP_MN_SET_POWER:
  804. Print(DBG_POWER_INFO, ("Power Setting %s state to %d\n",
  805. ((powerType == SystemPowerState) ? "System"
  806. : "Device"),
  807. powerState.SystemState));
  808. break;
  809. case IRP_MN_QUERY_POWER:
  810. Print (DBG_POWER_INFO, ("Power query %s status to %d\n",
  811. ((powerType == SystemPowerState) ? "System"
  812. : "Device"),
  813. powerState.SystemState));
  814. break;
  815. default:
  816. Print (DBG_POWER_ERROR, ("Power minor (0x%x) no known\n",
  817. stack->MinorFunction));
  818. }
  819. PoStartNextPowerIrp (Irp);
  820. IoSkipCurrentIrpStackLocation (Irp);
  821. status = PoCallDriver (data->TopOfStack, Irp);
  822. IoReleaseRemoveLock (&data->RemoveLock, Irp);
  823. return status;
  824. }