1 /*
2 * PROJECT: ReactOS USB Port Driver
3 * LICENSE: GPL-2.0+ (https://spdx.org/licenses/GPL-2.0+)
4 * PURPOSE: USBPort interface functions
5 * COPYRIGHT: Copyright 2017 Vadim Galyant <vgal@rambler.ru>
6 */
7
8 #include "usbport.h"
9
10 #define NDEBUG
11 #include <debug.h>
12
13 VOID
14 USB_BUSIFFN
USBI_InterfaceReference(IN PVOID BusContext)15 USBI_InterfaceReference(IN PVOID BusContext)
16 {
17 DPRINT("USBI_InterfaceReference\n");
18 }
19
20 VOID
21 USB_BUSIFFN
USBI_InterfaceDereference(IN PVOID BusContext)22 USBI_InterfaceDereference(IN PVOID BusContext)
23 {
24 DPRINT("USBI_InterfaceDereference\n");
25 }
26
27 /* USB port driver Interface functions */
28
29 NTSTATUS
30 USB_BUSIFFN
USBHI_CreateUsbDevice(IN PVOID BusContext,IN OUT PUSB_DEVICE_HANDLE * UsbdDeviceHandle,IN PUSB_DEVICE_HANDLE UsbdHubDeviceHandle,IN USHORT PortStatus,IN USHORT PortNumber)31 USBHI_CreateUsbDevice(IN PVOID BusContext,
32 IN OUT PUSB_DEVICE_HANDLE *UsbdDeviceHandle,
33 IN PUSB_DEVICE_HANDLE UsbdHubDeviceHandle,
34 IN USHORT PortStatus,
35 IN USHORT PortNumber)
36 {
37 PDEVICE_OBJECT PdoDevice;
38 PUSBPORT_RHDEVICE_EXTENSION PdoExtension;
39 PUSB_DEVICE_HANDLE deviceHandle = NULL;
40 NTSTATUS Status;
41
42 DPRINT("USBHI_CreateUsbDevice: ...\n");
43
44 PdoDevice = BusContext;
45 PdoExtension = PdoDevice->DeviceExtension;
46
47 Status = USBPORT_CreateDevice(&deviceHandle,
48 PdoExtension->FdoDevice,
49 (PUSBPORT_DEVICE_HANDLE)UsbdHubDeviceHandle,
50 PortStatus,
51 PortNumber);
52
53 *UsbdDeviceHandle = deviceHandle;
54
55 return Status;
56 }
57
58 NTSTATUS
59 USB_BUSIFFN
USBHI_InitializeUsbDevice(IN PVOID BusContext,OUT PUSB_DEVICE_HANDLE UsbdDeviceHandle)60 USBHI_InitializeUsbDevice(IN PVOID BusContext,
61 OUT PUSB_DEVICE_HANDLE UsbdDeviceHandle)
62 {
63 PDEVICE_OBJECT PdoDevice;
64 PUSBPORT_RHDEVICE_EXTENSION PdoExtension;
65
66 DPRINT("USBHI_InitializeUsbDevice\n");
67
68 PdoDevice = BusContext;
69 PdoExtension = PdoDevice->DeviceExtension;
70
71 return USBPORT_InitializeDevice((PUSBPORT_DEVICE_HANDLE)UsbdDeviceHandle,
72 PdoExtension->FdoDevice);
73 }
74
75 NTSTATUS
76 USB_BUSIFFN
USBHI_GetUsbDescriptors(IN PVOID BusContext,IN PUSB_DEVICE_HANDLE UsbdDeviceHandle,IN PUCHAR DeviceDescBuffer,IN PULONG DeviceDescBufferLen,IN PUCHAR ConfigDescBuffer,IN PULONG ConfigDescBufferLen)77 USBHI_GetUsbDescriptors(IN PVOID BusContext,
78 IN PUSB_DEVICE_HANDLE UsbdDeviceHandle,
79 IN PUCHAR DeviceDescBuffer,
80 IN PULONG DeviceDescBufferLen,
81 IN PUCHAR ConfigDescBuffer,
82 IN PULONG ConfigDescBufferLen)
83 {
84 PDEVICE_OBJECT PdoDevice;
85 PUSBPORT_RHDEVICE_EXTENSION PdoExtension;
86 PUSBPORT_DEVICE_HANDLE DeviceHandle;
87
88 NTSTATUS Status;
89
90 DPRINT("USBHI_GetUsbDescriptors ...\n");
91
92 PdoDevice = BusContext;
93 PdoExtension = PdoDevice->DeviceExtension;
94 DeviceHandle = (PUSBPORT_DEVICE_HANDLE)UsbdDeviceHandle;
95
96 if (DeviceDescBuffer && *DeviceDescBufferLen)
97 {
98 if (*DeviceDescBufferLen > sizeof(USB_DEVICE_DESCRIPTOR))
99 *DeviceDescBufferLen = sizeof(USB_DEVICE_DESCRIPTOR);
100
101 RtlCopyMemory(DeviceDescBuffer,
102 &DeviceHandle->DeviceDescriptor,
103 *DeviceDescBufferLen);
104 }
105
106 Status = USBPORT_GetUsbDescriptor(DeviceHandle,
107 PdoExtension->FdoDevice,
108 USB_CONFIGURATION_DESCRIPTOR_TYPE,
109 ConfigDescBuffer,
110 ConfigDescBufferLen);
111
112 USBPORT_DumpingDeviceDescriptor((PUSB_DEVICE_DESCRIPTOR)DeviceDescBuffer);
113
114 return Status;
115 }
116
117 NTSTATUS
118 USB_BUSIFFN
USBHI_RemoveUsbDevice(IN PVOID BusContext,IN OUT PUSB_DEVICE_HANDLE UsbdDeviceHandle,IN ULONG Flags)119 USBHI_RemoveUsbDevice(IN PVOID BusContext,
120 IN OUT PUSB_DEVICE_HANDLE UsbdDeviceHandle,
121 IN ULONG Flags)
122 {
123 PDEVICE_OBJECT PdoDevice;
124 PUSBPORT_RHDEVICE_EXTENSION PdoExtension;
125
126 DPRINT("USBHI_RemoveUsbDevice: UsbdDeviceHandle - %p, Flags - %x\n",
127 UsbdDeviceHandle,
128 Flags);
129
130 PdoDevice = BusContext;
131 PdoExtension = PdoDevice->DeviceExtension;
132
133 return USBPORT_RemoveDevice(PdoExtension->FdoDevice,
134 (PUSBPORT_DEVICE_HANDLE)UsbdDeviceHandle,
135 Flags);
136 }
137
138 NTSTATUS
139 USB_BUSIFFN
USBHI_RestoreUsbDevice(IN PVOID BusContext,OUT PUSB_DEVICE_HANDLE OldUsbdDeviceHandle,OUT PUSB_DEVICE_HANDLE NewUsbdDeviceHandle)140 USBHI_RestoreUsbDevice(IN PVOID BusContext,
141 OUT PUSB_DEVICE_HANDLE OldUsbdDeviceHandle,
142 OUT PUSB_DEVICE_HANDLE NewUsbdDeviceHandle)
143 {
144 PDEVICE_OBJECT PdoDevice;
145 PUSBPORT_RHDEVICE_EXTENSION PdoExtension;
146
147 DPRINT("USBHI_RestoreUsbDevice: OldUsbdDeviceHandle - %p, NewUsbdDeviceHandle - %x\n",
148 OldUsbdDeviceHandle,
149 NewUsbdDeviceHandle);
150
151 PdoDevice = BusContext;
152 PdoExtension = PdoDevice->DeviceExtension;
153
154 return USBPORT_RestoreDevice(PdoExtension->FdoDevice,
155 (PUSBPORT_DEVICE_HANDLE)OldUsbdDeviceHandle,
156 (PUSBPORT_DEVICE_HANDLE)NewUsbdDeviceHandle);
157 }
158
159 NTSTATUS
160 USB_BUSIFFN
USBHI_QueryDeviceInformation(IN PVOID BusContext,IN PUSB_DEVICE_HANDLE UsbdDeviceHandle,OUT PVOID DeviceInfoBuffer,IN ULONG DeviceInfoBufferLen,OUT PULONG LenDataReturned)161 USBHI_QueryDeviceInformation(IN PVOID BusContext,
162 IN PUSB_DEVICE_HANDLE UsbdDeviceHandle,
163 OUT PVOID DeviceInfoBuffer,
164 IN ULONG DeviceInfoBufferLen,
165 OUT PULONG LenDataReturned)
166 {
167 PUSB_DEVICE_INFORMATION_0 DeviceInfo;
168 PUSBPORT_CONFIGURATION_HANDLE ConfigHandle;
169 PLIST_ENTRY InterfaceEntry;
170 PUSBPORT_DEVICE_HANDLE DeviceHandle;
171 ULONG NumberOfOpenPipes = 0;
172 PUSB_PIPE_INFORMATION_0 PipeInfo;
173 PUSBPORT_PIPE_HANDLE PipeHandle;
174 PUSBPORT_INTERFACE_HANDLE InterfaceHandle;
175 ULONG ActualLength;
176 ULONG ix;
177
178 DPRINT("USBHI_QueryDeviceInformation: ...\n");
179
180 *LenDataReturned = 0;
181
182 if (DeviceInfoBufferLen < sizeof(USB_LEVEL_INFORMATION))
183 {
184 return STATUS_BUFFER_TOO_SMALL;
185 }
186
187 DeviceInfo = DeviceInfoBuffer;
188
189 if (DeviceInfo->InformationLevel > 0)
190 {
191 return STATUS_NOT_SUPPORTED;
192 }
193
194 DeviceHandle = UsbdDeviceHandle;
195 ConfigHandle = DeviceHandle->ConfigHandle;
196
197 if (ConfigHandle)
198 {
199 InterfaceEntry = ConfigHandle->InterfaceHandleList.Flink;
200
201 while (InterfaceEntry &&
202 InterfaceEntry != &ConfigHandle->InterfaceHandleList)
203 {
204 InterfaceHandle = CONTAINING_RECORD(InterfaceEntry,
205 USBPORT_INTERFACE_HANDLE,
206 InterfaceLink);
207
208 NumberOfOpenPipes += InterfaceHandle->InterfaceDescriptor.bNumEndpoints;
209
210 InterfaceEntry = InterfaceEntry->Flink;
211 }
212 }
213
214 ActualLength = FIELD_OFFSET(USB_DEVICE_INFORMATION_0, PipeList) +
215 NumberOfOpenPipes * sizeof(USB_PIPE_INFORMATION_0);
216
217 if (DeviceInfoBufferLen < ActualLength)
218 {
219 DeviceInfo->ActualLength = ActualLength;
220 *LenDataReturned = sizeof(USB_LEVEL_INFORMATION);
221
222 return STATUS_BUFFER_TOO_SMALL;
223 }
224
225 RtlZeroMemory(DeviceInfo, ActualLength);
226
227 DeviceInfo->InformationLevel = 0;
228 DeviceInfo->ActualLength = ActualLength;
229 DeviceInfo->DeviceAddress = DeviceHandle->DeviceAddress;
230 DeviceInfo->NumberOfOpenPipes = NumberOfOpenPipes;
231 DeviceInfo->DeviceSpeed = DeviceHandle->DeviceSpeed;
232
233 RtlCopyMemory(&DeviceInfo->DeviceDescriptor,
234 &DeviceHandle->DeviceDescriptor,
235 sizeof(USB_DEVICE_DESCRIPTOR));
236
237 USBPORT_DumpingDeviceDescriptor(&DeviceInfo->DeviceDescriptor);
238
239 if (DeviceHandle->DeviceSpeed == UsbFullSpeed ||
240 DeviceHandle->DeviceSpeed == UsbLowSpeed)
241 {
242 DeviceInfo->DeviceType = Usb11Device;
243 }
244 else if (DeviceHandle->DeviceSpeed == UsbHighSpeed)
245 {
246 DeviceInfo->DeviceType = Usb20Device;
247 }
248
249 DeviceInfo->CurrentConfigurationValue = 0;
250
251 if (!ConfigHandle)
252 {
253 *LenDataReturned = ActualLength;
254 return STATUS_SUCCESS;
255 }
256
257 DeviceInfo->CurrentConfigurationValue =
258 ConfigHandle->ConfigurationDescriptor->bConfigurationValue;
259
260 InterfaceEntry = ConfigHandle->InterfaceHandleList.Flink;
261
262 while (InterfaceEntry &&
263 InterfaceEntry != &ConfigHandle->InterfaceHandleList)
264 {
265 InterfaceHandle = CONTAINING_RECORD(InterfaceEntry,
266 USBPORT_INTERFACE_HANDLE,
267 InterfaceLink);
268
269 if (InterfaceHandle->InterfaceDescriptor.bNumEndpoints > 0)
270 {
271 PipeInfo = &DeviceInfo->PipeList[0];
272 PipeHandle = &InterfaceHandle->PipeHandle[0];
273
274 for (ix = 0;
275 ix < InterfaceHandle->InterfaceDescriptor.bNumEndpoints;
276 ix++)
277 {
278 if (PipeHandle->Flags & PIPE_HANDLE_FLAG_NULL_PACKET_SIZE)
279 {
280 PipeInfo->ScheduleOffset = 1;
281 }
282 else
283 {
284 PipeInfo->ScheduleOffset =
285 PipeHandle->Endpoint->EndpointProperties.ScheduleOffset;
286 }
287
288 RtlCopyMemory(&PipeInfo->EndpointDescriptor,
289 &PipeHandle->EndpointDescriptor,
290 sizeof(USB_ENDPOINT_DESCRIPTOR));
291
292 PipeInfo += 1;
293 PipeHandle += 1;
294 }
295 }
296
297 InterfaceEntry = InterfaceEntry->Flink;
298 }
299
300 *LenDataReturned = ActualLength;
301
302 return STATUS_SUCCESS;
303 }
304
305 NTSTATUS
306 USB_BUSIFFN
USBHI_GetControllerInformation(IN PVOID BusContext,OUT PVOID ControllerInfoBuffer,IN ULONG ControllerInfoBufferLen,OUT PULONG LenDataReturned)307 USBHI_GetControllerInformation(IN PVOID BusContext,
308 OUT PVOID ControllerInfoBuffer,
309 IN ULONG ControllerInfoBufferLen,
310 OUT PULONG LenDataReturned)
311 {
312 PDEVICE_OBJECT PdoDevice;
313 PUSBPORT_RHDEVICE_EXTENSION PdoExtension;
314 PDEVICE_OBJECT FdoDevice;
315 PUSBPORT_DEVICE_EXTENSION FdoExtension;
316 PUSB_CONTROLLER_INFORMATION_0 InfoBuffer;
317 NTSTATUS Status;
318
319 DPRINT("USBHI_GetControllerInformation: ControllerInfoBufferLen - %x\n",
320 ControllerInfoBufferLen);
321
322 PdoDevice = BusContext;
323 PdoExtension = PdoDevice->DeviceExtension;
324 FdoDevice = PdoExtension->FdoDevice;
325 FdoExtension = FdoDevice->DeviceExtension;
326
327 InfoBuffer = ControllerInfoBuffer;
328
329 *LenDataReturned = 0;
330
331 if (ControllerInfoBufferLen < sizeof(USB_LEVEL_INFORMATION))
332 {
333 Status = STATUS_BUFFER_TOO_SMALL;
334 return Status;
335 }
336
337 *LenDataReturned = sizeof(USB_LEVEL_INFORMATION);
338
339 if (InfoBuffer->InformationLevel > 0)
340 {
341 Status = STATUS_NOT_SUPPORTED;
342 return Status;
343 }
344
345 InfoBuffer->ActualLength = sizeof(USB_CONTROLLER_INFORMATION_0);
346
347 if (ControllerInfoBufferLen >= sizeof(USB_CONTROLLER_INFORMATION_0))
348 {
349 InfoBuffer->SelectiveSuspendEnabled =
350 (FdoExtension->Flags & USBPORT_FLAG_SELECTIVE_SUSPEND) ==
351 USBPORT_FLAG_SELECTIVE_SUSPEND;
352 }
353
354 *LenDataReturned = sizeof(USB_CONTROLLER_INFORMATION_0);
355
356 return STATUS_SUCCESS;
357 }
358
359 NTSTATUS
360 USB_BUSIFFN
USBHI_ControllerSelectiveSuspend(IN PVOID BusContext,IN BOOLEAN Enable)361 USBHI_ControllerSelectiveSuspend(IN PVOID BusContext,
362 IN BOOLEAN Enable)
363 {
364 PDEVICE_OBJECT PdoDevice;
365 PUSBPORT_RHDEVICE_EXTENSION PdoExtension;
366 PDEVICE_OBJECT FdoDevice;
367 PUSBPORT_DEVICE_EXTENSION FdoExtension;
368 ULONG Flags;
369 ULONG HcDisable;
370 NTSTATUS Status;
371
372 DPRINT("USBHI_ControllerSelectiveSuspend: Enable - %x\n", Enable);
373
374 PdoDevice = BusContext;
375 PdoExtension = PdoDevice->DeviceExtension;
376 FdoDevice = PdoExtension->FdoDevice;
377 FdoExtension = FdoDevice->DeviceExtension;
378
379 Flags = FdoExtension->Flags;
380
381 if (Flags & USBPORT_FLAG_BIOS_DISABLE_SS)
382 {
383 return STATUS_SUCCESS;
384 }
385
386 if (Enable)
387 {
388 FdoExtension->Flags |= USBPORT_FLAG_SELECTIVE_SUSPEND;
389 HcDisable = 0;
390 }
391 else
392 {
393 FdoExtension->Flags &= ~USBPORT_FLAG_SELECTIVE_SUSPEND;
394 HcDisable = 1;
395 }
396
397 Status = USBPORT_SetRegistryKeyValue(FdoExtension->CommonExtension.LowerPdoDevice,
398 TRUE,
399 REG_DWORD,
400 L"HcDisableSelectiveSuspend",
401 &HcDisable,
402 sizeof(HcDisable));
403
404 if (NT_SUCCESS(Status))
405 {
406 if (Enable)
407 FdoExtension->Flags |= USBPORT_FLAG_SELECTIVE_SUSPEND;
408 else
409 FdoExtension->Flags &= ~USBPORT_FLAG_SELECTIVE_SUSPEND;
410 }
411
412 return Status;
413 }
414
415 NTSTATUS
416 USB_BUSIFFN
USBHI_GetExtendedHubInformation(IN PVOID BusContext,IN PDEVICE_OBJECT HubPhysicalDeviceObject,IN OUT PVOID HubInformationBuffer,IN ULONG HubInfoLen,IN OUT PULONG LenDataReturned)417 USBHI_GetExtendedHubInformation(IN PVOID BusContext,
418 IN PDEVICE_OBJECT HubPhysicalDeviceObject,
419 IN OUT PVOID HubInformationBuffer,
420 IN ULONG HubInfoLen,
421 IN OUT PULONG LenDataReturned)
422 {
423 PDEVICE_OBJECT PdoDevice;
424 PUSBPORT_RHDEVICE_EXTENSION PdoExtension;
425 PDEVICE_OBJECT FdoDevice;
426 PUSBPORT_DEVICE_EXTENSION FdoExtension;
427 PUSBPORT_REGISTRATION_PACKET Packet;
428 ULONG NumPorts;
429 ULONG ix;
430 PUSB_EXTHUB_INFORMATION_0 HubInfoBuffer;
431 USB_PORT_STATUS_AND_CHANGE PortStatus;
432 ULONG PortAttrX;
433
434 DPRINT("USBHI_GetExtendedHubInformation: ...\n");
435
436 PdoDevice = BusContext;
437 PdoExtension = PdoDevice->DeviceExtension;
438 FdoDevice = PdoExtension->FdoDevice;
439 FdoExtension = FdoDevice->DeviceExtension;
440 Packet = &FdoExtension->MiniPortInterface->Packet;
441
442 HubInfoBuffer = HubInformationBuffer;
443 PortStatus.AsUlong32 = 0;
444
445 if (HubPhysicalDeviceObject != PdoDevice)
446 {
447 *LenDataReturned = 0;
448 return STATUS_NOT_SUPPORTED;
449 }
450
451 if (HubInfoLen < sizeof(USB_EXTHUB_INFORMATION_0))
452 {
453 *LenDataReturned = 0;
454 return STATUS_BUFFER_TOO_SMALL;
455 }
456
457 NumPorts = PdoExtension->RootHubDescriptors->Descriptor.bNumberOfPorts;
458 HubInfoBuffer->NumberOfPorts = NumPorts;
459
460 if (NumPorts == 0)
461 {
462 *LenDataReturned = sizeof(USB_EXTHUB_INFORMATION_0);
463 return STATUS_SUCCESS;
464 }
465
466 for (ix = 0; ix < HubInfoBuffer->NumberOfPorts; ++ix)
467 {
468 HubInfoBuffer->Port[ix].PhysicalPortNumber = ix + 1;
469 HubInfoBuffer->Port[ix].PortLabelNumber = ix;
470 HubInfoBuffer->Port[ix].VidOverride = 0;
471 HubInfoBuffer->Port[ix].PidOverride = 0;
472 HubInfoBuffer->Port[ix].PortAttributes = 0;
473
474 if (Packet->MiniPortFlags & USB_MINIPORT_FLAGS_USB2)
475 {
476 HubInfoBuffer->Port[ix].PortAttributes = USB_PORTATTR_SHARED_USB2;
477
478 Packet->RH_GetPortStatus(FdoExtension->MiniPortExt,
479 ix + 1,
480 &PortStatus);
481
482 if (PortStatus.PortStatus.Usb20PortStatus.AsUshort16 & 0x8000)
483 {
484 HubInfoBuffer->Port[ix].PortAttributes |= USB_PORTATTR_OWNED_BY_CC;
485 }
486 }
487 else
488 {
489 if (!(FdoExtension->Flags & USBPORT_FLAG_COMPANION_HC))
490 {
491 continue;
492 }
493
494 if (USBPORT_FindUSB2Controller(FdoDevice))
495 {
496 HubInfoBuffer->Port[ix].PortAttributes |= USB_PORTATTR_NO_OVERCURRENT_UI;
497 }
498 }
499 }
500
501 for (ix = 0; ix < HubInfoBuffer->NumberOfPorts; ++ix)
502 {
503 PortAttrX = 0;
504
505 USBPORT_GetRegistryKeyValueFullInfo(FdoDevice,
506 FdoExtension->CommonExtension.LowerPdoDevice,
507 FALSE,
508 L"PortAttrX",
509 sizeof(L"PortAttrX"),
510 &PortAttrX,
511 sizeof(PortAttrX));
512
513 HubInfoBuffer->Port[ix].PortAttributes |= PortAttrX;
514 }
515
516 *LenDataReturned = sizeof(USB_EXTHUB_INFORMATION_0);
517
518 return STATUS_SUCCESS;
519 }
520
521 NTSTATUS
522 USB_BUSIFFN
USBHI_GetRootHubSymbolicName(IN PVOID BusContext,IN OUT PVOID HubInfoBuffer,IN ULONG HubInfoBufferLen,OUT PULONG HubNameActualLen)523 USBHI_GetRootHubSymbolicName(IN PVOID BusContext,
524 IN OUT PVOID HubInfoBuffer,
525 IN ULONG HubInfoBufferLen,
526 OUT PULONG HubNameActualLen)
527 {
528 PDEVICE_OBJECT PdoDevice;
529 UNICODE_STRING HubName;
530 PUNICODE_STRING InfoBuffer;
531 NTSTATUS Status;
532
533 DPRINT("USBHI_GetRootHubSymbolicName: ...\n");
534
535 PdoDevice = BusContext;
536
537 Status = USBPORT_GetSymbolicName(PdoDevice, &HubName);
538
539 if (HubInfoBufferLen < HubName.Length)
540 {
541 InfoBuffer = HubInfoBuffer;
542 InfoBuffer->Length = 0;
543 }
544 else
545 {
546 RtlCopyMemory(HubInfoBuffer, HubName.Buffer, HubName.Length);
547 }
548
549 *HubNameActualLen = HubName.Length;
550
551 if (NT_SUCCESS(Status))
552 RtlFreeUnicodeString(&HubName);
553
554 return Status;
555 }
556
557 PVOID
558 USB_BUSIFFN
USBHI_GetDeviceBusContext(IN PVOID BusContext,IN PVOID DeviceHandle)559 USBHI_GetDeviceBusContext(IN PVOID BusContext,
560 IN PVOID DeviceHandle)
561 {
562 DPRINT1("USBHI_GetDeviceBusContext: UNIMPLEMENTED. FIXME.\n");
563 return NULL;
564 }
565
566 NTSTATUS
567 USB_BUSIFFN
USBHI_Initialize20Hub(IN PVOID BusContext,IN PUSB_DEVICE_HANDLE UsbdHubDeviceHandle,IN ULONG TtCount)568 USBHI_Initialize20Hub(IN PVOID BusContext,
569 IN PUSB_DEVICE_HANDLE UsbdHubDeviceHandle,
570 IN ULONG TtCount)
571 {
572 PDEVICE_OBJECT PdoDevice;
573 PUSBPORT_RHDEVICE_EXTENSION PdoExtension;
574
575 DPRINT("USBHI_Initialize20Hub: UsbdHubDeviceHandle - %p, TtCount - %x\n",
576 UsbdHubDeviceHandle,
577 TtCount);
578
579 PdoDevice = BusContext;
580 PdoExtension = PdoDevice->DeviceExtension;
581
582 return USBPORT_Initialize20Hub(PdoExtension->FdoDevice,
583 (PUSBPORT_DEVICE_HANDLE)UsbdHubDeviceHandle,
584 TtCount);
585 }
586
587 NTSTATUS
588 USB_BUSIFFN
USBHI_RootHubInitNotification(IN PVOID BusContext,IN PVOID CallbackContext,IN PRH_INIT_CALLBACK CallbackFunction)589 USBHI_RootHubInitNotification(IN PVOID BusContext,
590 IN PVOID CallbackContext,
591 IN PRH_INIT_CALLBACK CallbackFunction)
592 {
593 PDEVICE_OBJECT PdoDevice;
594 PUSBPORT_RHDEVICE_EXTENSION PdoExtension;
595 PDEVICE_OBJECT FdoDevice;
596 PUSBPORT_DEVICE_EXTENSION FdoExtension;
597 KIRQL OldIrql;
598
599 DPRINT("USBHI_RootHubInitNotification\n");
600
601 PdoDevice = BusContext;
602 PdoExtension = PdoDevice->DeviceExtension;
603 FdoDevice = PdoExtension->FdoDevice;
604 FdoExtension = FdoDevice->DeviceExtension;
605
606 KeAcquireSpinLock(&FdoExtension->RootHubCallbackSpinLock, &OldIrql);
607 PdoExtension->RootHubInitContext = CallbackContext;
608 PdoExtension->RootHubInitCallback = CallbackFunction;
609 KeReleaseSpinLock(&FdoExtension->RootHubCallbackSpinLock, OldIrql);
610
611 return STATUS_SUCCESS;
612 }
613
614 VOID
615 USB_BUSIFFN
USBHI_FlushTransfers(IN PVOID BusContext,OUT PUSB_DEVICE_HANDLE UsbdDeviceHandle)616 USBHI_FlushTransfers(IN PVOID BusContext,
617 OUT PUSB_DEVICE_HANDLE UsbdDeviceHandle)
618 {
619 PDEVICE_OBJECT PdoDevice;
620 PUSBPORT_RHDEVICE_EXTENSION PdoExtension;
621
622 DPRINT("USBHI_FlushTransfers: ...\n");
623
624 PdoDevice = BusContext;
625 PdoExtension = PdoDevice->DeviceExtension;
626
627 USBPORT_BadRequestFlush(PdoExtension->FdoDevice);
628 }
629
630 VOID
631 USB_BUSIFFN
USBHI_SetDeviceHandleData(IN PVOID BusContext,IN PVOID DeviceHandle,IN PDEVICE_OBJECT UsbDevicePdo)632 USBHI_SetDeviceHandleData(IN PVOID BusContext,
633 IN PVOID DeviceHandle,
634 IN PDEVICE_OBJECT UsbDevicePdo)
635 {
636 DPRINT1("USBHI_SetDeviceHandleData: UNIMPLEMENTED. FIXME.\n");
637 }
638
639 /* USB bus driver Interface functions */
640
641 VOID
642 USB_BUSIFFN
USBDI_GetUSBDIVersion(IN PVOID BusContext,OUT PUSBD_VERSION_INFORMATION VersionInfo,OUT PULONG HcdCapabilities)643 USBDI_GetUSBDIVersion(IN PVOID BusContext,
644 OUT PUSBD_VERSION_INFORMATION VersionInfo,
645 OUT PULONG HcdCapabilities)
646 {
647 DPRINT1("USBDI_GetUSBDIVersion: UNIMPLEMENTED. FIXME.\n");
648 }
649
650 NTSTATUS
651 USB_BUSIFFN
USBDI_QueryBusTime(IN PVOID BusContext,OUT PULONG CurrentFrame)652 USBDI_QueryBusTime(IN PVOID BusContext,
653 OUT PULONG CurrentFrame)
654 {
655 DPRINT1("USBDI_QueryBusTime: UNIMPLEMENTED. FIXME.\n");
656 return STATUS_SUCCESS;
657 }
658
659 NTSTATUS
660 USB_BUSIFFN
USBDI_SubmitIsoOutUrb(IN PVOID BusContext,IN PURB Urb)661 USBDI_SubmitIsoOutUrb(IN PVOID BusContext,
662 IN PURB Urb)
663 {
664 DPRINT1("USBDI_SubmitIsoOutUrb: UNIMPLEMENTED. FIXME.\n");
665 return STATUS_SUCCESS;
666 }
667
668 NTSTATUS
669 USB_BUSIFFN
USBDI_QueryBusInformation(IN PVOID BusContext,IN ULONG Level,OUT PVOID BusInfoBuffer,OUT PULONG BusInfoBufferLen,OUT PULONG BusInfoActualLen)670 USBDI_QueryBusInformation(IN PVOID BusContext,
671 IN ULONG Level,
672 OUT PVOID BusInfoBuffer,
673 OUT PULONG BusInfoBufferLen,
674 OUT PULONG BusInfoActualLen)
675 {
676 PDEVICE_OBJECT PdoDevice;
677 PUSBPORT_RHDEVICE_EXTENSION PdoExtension;
678 PDEVICE_OBJECT FdoDevice;
679 PUSBPORT_DEVICE_EXTENSION FdoExtension;
680 SIZE_T Length;
681 PUSB_BUS_INFORMATION_LEVEL_1 Buffer1;
682
683 DPRINT("USBDI_QueryBusInformation: Level - %p\n", Level);
684
685 if ((Level != 0) && (Level != 1))
686 {
687 DPRINT1("USBDI_QueryBusInformation: Level should be 0 or 1\n");
688 return STATUS_NOT_SUPPORTED;
689 }
690
691 PdoDevice = BusContext;
692 PdoExtension = PdoDevice->DeviceExtension;
693 FdoDevice = PdoExtension->FdoDevice;
694 FdoExtension = FdoDevice->DeviceExtension;
695
696 if (Level == 0)
697 {
698 if (BusInfoActualLen)
699 *BusInfoActualLen = sizeof(USB_BUS_INFORMATION_LEVEL_0);
700
701 if (*BusInfoBufferLen < sizeof(USB_BUS_INFORMATION_LEVEL_0))
702 {
703 return STATUS_BUFFER_TOO_SMALL;
704 }
705
706 *BusInfoBufferLen = sizeof(USB_BUS_INFORMATION_LEVEL_0);
707
708 //Buffer0 = BusInfoBuffer;
709 DPRINT1("USBDI_QueryBusInformation: LEVEL_0 UNIMPLEMENTED. FIXME\n");
710 //Buffer0->TotalBandwidth = USBPORT_GetTotalBandwidth();
711 //Buffer0->ConsumedBandwidth = USBPORT_GetAllocatedBandwidth();
712
713 return STATUS_SUCCESS;
714 }
715
716 if (Level == 1)
717 {
718 Length = sizeof(USB_BUS_INFORMATION_LEVEL_1) +
719 FdoExtension->CommonExtension.SymbolicLinkName.Length;
720
721 if (BusInfoActualLen)
722 *BusInfoActualLen = Length;
723
724 if (*BusInfoBufferLen < Length)
725 {
726 return STATUS_BUFFER_TOO_SMALL;
727 }
728
729 *BusInfoBufferLen = Length;
730
731 Buffer1 = BusInfoBuffer;
732 DPRINT1("USBDI_QueryBusInformation: LEVEL_1 UNIMPLEMENTED. FIXME\n");
733 //Buffer1->TotalBandwidth = USBPORT_GetTotalBandwidth();
734 //Buffer1->ConsumedBandwidth = USBPORT_GetAllocatedBandwidth();
735 Buffer1->ControllerNameLength = FdoExtension->CommonExtension.SymbolicLinkName.Length;
736
737 RtlCopyMemory(&Buffer1->ControllerNameUnicodeString,
738 FdoExtension->CommonExtension.SymbolicLinkName.Buffer,
739 FdoExtension->CommonExtension.SymbolicLinkName.Length);
740
741 return STATUS_SUCCESS;
742 }
743
744 return STATUS_SUCCESS;
745 }
746
747 BOOLEAN
748 USB_BUSIFFN
USBDI_IsDeviceHighSpeed(IN PVOID BusContext)749 USBDI_IsDeviceHighSpeed(IN PVOID BusContext)
750 {
751 PDEVICE_OBJECT PdoDevice;
752 PUSBPORT_RHDEVICE_EXTENSION PdoExtension;
753 PDEVICE_OBJECT FdoDevice;
754 PUSBPORT_DEVICE_EXTENSION FdoExtension;
755 PUSBPORT_REGISTRATION_PACKET Packet;
756
757 DPRINT("USBDI_IsDeviceHighSpeed: ...\n");
758
759 PdoDevice = BusContext;
760 PdoExtension = PdoDevice->DeviceExtension;
761 FdoDevice = PdoExtension->FdoDevice;
762 FdoExtension = FdoDevice->DeviceExtension;
763 Packet = &FdoExtension->MiniPortInterface->Packet;
764
765 return (Packet->MiniPortFlags & USB_MINIPORT_FLAGS_USB2) != 0;
766 }
767
768 NTSTATUS
769 USB_BUSIFFN
USBDI_EnumLogEntry(IN PVOID BusContext,IN ULONG DriverTag,IN ULONG EnumTag,IN ULONG P1,IN ULONG P2)770 USBDI_EnumLogEntry(IN PVOID BusContext,
771 IN ULONG DriverTag,
772 IN ULONG EnumTag,
773 IN ULONG P1,
774 IN ULONG P2)
775 {
776 DPRINT1("USBDI_EnumLogEntry: UNIMPLEMENTED. FIXME.\n");
777 return STATUS_SUCCESS;
778 }
779
780 NTSTATUS
781 NTAPI
USBPORT_PdoQueryInterface(IN PDEVICE_OBJECT FdoDevice,IN PDEVICE_OBJECT PdoDevice,IN PIRP Irp)782 USBPORT_PdoQueryInterface(IN PDEVICE_OBJECT FdoDevice,
783 IN PDEVICE_OBJECT PdoDevice,
784 IN PIRP Irp)
785 {
786 PIO_STACK_LOCATION IoStack = IoGetCurrentIrpStackLocation(Irp);
787 PUSB_BUS_INTERFACE_HUB_V5 InterfaceHub;
788 PUSB_BUS_INTERFACE_USBDI_V2 InterfaceDI;
789 UNICODE_STRING GuidBuffer;
790 NTSTATUS Status;
791
792 DPRINT("USBPORT_PdoQueryInterface: ...\n");
793
794 if (IsEqualGUIDAligned(IoStack->Parameters.QueryInterface.InterfaceType,
795 &USB_BUS_INTERFACE_HUB_GUID))
796 {
797 /* Get request parameters */
798 InterfaceHub = (PUSB_BUS_INTERFACE_HUB_V5)IoStack->Parameters.QueryInterface.Interface;
799 InterfaceHub->Version = IoStack->Parameters.QueryInterface.Version;
800
801 /* Check version */
802 if (IoStack->Parameters.QueryInterface.Version >= 6)
803 {
804 DPRINT1("USB_BUS_INTERFACE_HUB_GUID version %x not supported!\n",
805 IoStack->Parameters.QueryInterface.Version);
806
807 return Irp->IoStatus.Status; // Version not supported
808 }
809
810 /* Interface version 0 */
811 InterfaceHub->Size = IoStack->Parameters.QueryInterface.Size;
812 InterfaceHub->BusContext = PdoDevice;
813
814 InterfaceHub->InterfaceReference = USBI_InterfaceReference;
815 InterfaceHub->InterfaceDereference = USBI_InterfaceDereference;
816
817 /* Interface version 1 */
818 if (IoStack->Parameters.QueryInterface.Version >= 1)
819 {
820 InterfaceHub->CreateUsbDevice = USBHI_CreateUsbDevice;
821 InterfaceHub->InitializeUsbDevice = USBHI_InitializeUsbDevice;
822 InterfaceHub->GetUsbDescriptors = USBHI_GetUsbDescriptors;
823 InterfaceHub->RemoveUsbDevice = USBHI_RemoveUsbDevice;
824 InterfaceHub->RestoreUsbDevice = USBHI_RestoreUsbDevice;
825 InterfaceHub->QueryDeviceInformation = USBHI_QueryDeviceInformation;
826 }
827
828 /* Interface version 2 */
829 if (IoStack->Parameters.QueryInterface.Version >= 2)
830 {
831 InterfaceHub->GetControllerInformation = USBHI_GetControllerInformation;
832 InterfaceHub->ControllerSelectiveSuspend = USBHI_ControllerSelectiveSuspend;
833 InterfaceHub->GetExtendedHubInformation = USBHI_GetExtendedHubInformation;
834 InterfaceHub->GetRootHubSymbolicName = USBHI_GetRootHubSymbolicName;
835 InterfaceHub->GetDeviceBusContext = USBHI_GetDeviceBusContext;
836 InterfaceHub->Initialize20Hub = USBHI_Initialize20Hub;
837 }
838
839 /* Interface version 3 */
840 if (IoStack->Parameters.QueryInterface.Version >= 3)
841 InterfaceHub->RootHubInitNotification = USBHI_RootHubInitNotification;
842
843 /* Interface version 4 */
844 if (IoStack->Parameters.QueryInterface.Version >= 4)
845 InterfaceHub->FlushTransfers = USBHI_FlushTransfers;
846
847 /* Interface version 5 */
848 if (IoStack->Parameters.QueryInterface.Version >= 5)
849 InterfaceHub->SetDeviceHandleData = USBHI_SetDeviceHandleData;
850
851 /* Request completed */
852 return STATUS_SUCCESS;
853 }
854 else if (IsEqualGUIDAligned(IoStack->Parameters.QueryInterface.InterfaceType,
855 &USB_BUS_INTERFACE_USBDI_GUID))
856 {
857 /* Get request parameters */
858 InterfaceDI = (PUSB_BUS_INTERFACE_USBDI_V2)IoStack->Parameters.QueryInterface.Interface;
859 InterfaceDI->Version = IoStack->Parameters.QueryInterface.Version;
860
861 /* Check version */
862 if (IoStack->Parameters.QueryInterface.Version >= 3)
863 {
864 DPRINT1("USB_BUS_INTERFACE_USBDI_GUID version %x not supported!\n",
865 IoStack->Parameters.QueryInterface.Version);
866
867 return Irp->IoStatus.Status; // Version not supported
868 }
869
870 /* Interface version 0 */
871 InterfaceDI->Size = IoStack->Parameters.QueryInterface.Size;
872 InterfaceDI->BusContext = PdoDevice;
873 InterfaceDI->InterfaceReference = USBI_InterfaceReference;
874 InterfaceDI->InterfaceDereference = USBI_InterfaceDereference;
875 InterfaceDI->GetUSBDIVersion = USBDI_GetUSBDIVersion;
876 InterfaceDI->QueryBusTime = USBDI_QueryBusTime;
877 InterfaceDI->SubmitIsoOutUrb = USBDI_SubmitIsoOutUrb;
878 InterfaceDI->QueryBusInformation = USBDI_QueryBusInformation;
879
880 /* Interface version 1 */
881 if (IoStack->Parameters.QueryInterface.Version >= 1)
882 InterfaceDI->IsDeviceHighSpeed = USBDI_IsDeviceHighSpeed;
883
884 /* Interface version 2 */
885 if (IoStack->Parameters.QueryInterface.Version >= 2)
886 InterfaceDI->EnumLogEntry = USBDI_EnumLogEntry;
887
888 return STATUS_SUCCESS;
889 }
890 else
891 {
892 /* Convert GUID to string */
893 Status = RtlStringFromGUID(IoStack->Parameters.QueryInterface.InterfaceType,
894 &GuidBuffer);
895
896 if (NT_SUCCESS(Status))
897 {
898 /* Print interface */
899 DPRINT1("HandleQueryInterface UNKNOWN INTERFACE GUID: %wZ Version %x\n",
900 &GuidBuffer,
901 IoStack->Parameters.QueryInterface.Version);
902
903 RtlFreeUnicodeString(&GuidBuffer); // Free GUID buffer
904 }
905 }
906
907 return Irp->IoStatus.Status;
908 }
909