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 15 USBI_InterfaceReference(IN PVOID BusContext) 16 { 17 DPRINT("USBI_InterfaceReference\n"); 18 } 19 20 VOID 21 USB_BUSIFFN 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 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 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 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 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 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 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 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 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 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 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 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 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 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 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 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 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 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 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 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 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 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 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