1c2c66affSColin Finck /* 2c2c66affSColin Finck * PROJECT: ReactOS USB Port Driver 3c2c66affSColin Finck * LICENSE: GPL-2.0+ (https://spdx.org/licenses/GPL-2.0+) 4c2c66affSColin Finck * PURPOSE: USBPort device functions 5c2c66affSColin Finck * COPYRIGHT: Copyright 2017 Vadim Galyant <vgal@rambler.ru> 6c2c66affSColin Finck */ 7c2c66affSColin Finck 8c2c66affSColin Finck #include "usbport.h" 9c2c66affSColin Finck 10c2c66affSColin Finck #define NDEBUG 11c2c66affSColin Finck #include <debug.h> 12c2c66affSColin Finck 13c2c66affSColin Finck NTSTATUS 14c2c66affSColin Finck NTAPI 15c2c66affSColin Finck USBPORT_SendSetupPacket(IN PUSBPORT_DEVICE_HANDLE DeviceHandle, 16c2c66affSColin Finck IN PDEVICE_OBJECT FdoDevice, 17c2c66affSColin Finck IN PUSB_DEFAULT_PIPE_SETUP_PACKET SetupPacket, 18c2c66affSColin Finck IN PVOID Buffer, 19c2c66affSColin Finck IN ULONG Length, 20c2c66affSColin Finck IN OUT PULONG TransferedLen, 21c2c66affSColin Finck IN OUT PUSBD_STATUS pUSBDStatus) 22c2c66affSColin Finck { 23c2c66affSColin Finck PURB Urb; 24c2c66affSColin Finck PMDL Mdl; 25c2c66affSColin Finck USBD_STATUS USBDStatus; 26c2c66affSColin Finck KEVENT Event; 27c2c66affSColin Finck NTSTATUS Status; 28c2c66affSColin Finck 29c2c66affSColin Finck DPRINT("USBPORT_SendSetupPacket: DeviceHandle - %p, FdoDevice - %p, SetupPacket - %p, Buffer - %p, Length - %x, TransferedLen - %x, pUSBDStatus - %x\n", 30c2c66affSColin Finck DeviceHandle, 31c2c66affSColin Finck FdoDevice, 32c2c66affSColin Finck SetupPacket, 33c2c66affSColin Finck Buffer, 34c2c66affSColin Finck Length, 35c2c66affSColin Finck TransferedLen, 36c2c66affSColin Finck pUSBDStatus); 37c2c66affSColin Finck 38c2c66affSColin Finck KeInitializeEvent(&Event, NotificationEvent, FALSE); 39c2c66affSColin Finck 40c2c66affSColin Finck Urb = ExAllocatePoolWithTag(NonPagedPool, 41c2c66affSColin Finck sizeof(struct _URB_CONTROL_TRANSFER), 42c2c66affSColin Finck USB_PORT_TAG); 43c2c66affSColin Finck 44c2c66affSColin Finck if (Urb) 45c2c66affSColin Finck { 46c2c66affSColin Finck InterlockedIncrement(&DeviceHandle->DeviceHandleLock); 47c2c66affSColin Finck 48c2c66affSColin Finck RtlZeroMemory(Urb, sizeof(struct _URB_CONTROL_TRANSFER)); 49c2c66affSColin Finck 50c2c66affSColin Finck RtlCopyMemory(Urb->UrbControlTransfer.SetupPacket, 51c2c66affSColin Finck SetupPacket, 52c2c66affSColin Finck sizeof(USB_DEFAULT_PIPE_SETUP_PACKET)); 53c2c66affSColin Finck 54c2c66affSColin Finck Urb->UrbHeader.Length = sizeof(struct _URB_CONTROL_TRANSFER); 55c2c66affSColin Finck Urb->UrbHeader.Function = URB_FUNCTION_CONTROL_TRANSFER; 56c2c66affSColin Finck Urb->UrbHeader.UsbdDeviceHandle = DeviceHandle; 57c2c66affSColin Finck Urb->UrbHeader.UsbdFlags = 0; 58c2c66affSColin Finck 59c2c66affSColin Finck Urb->UrbControlTransfer.PipeHandle = &DeviceHandle->PipeHandle; 60c2c66affSColin Finck Urb->UrbControlTransfer.TransferBufferLength = Length; 61c2c66affSColin Finck Urb->UrbControlTransfer.TransferBuffer = Buffer; 62c2c66affSColin Finck Urb->UrbControlTransfer.TransferBufferMDL = NULL; 63c2c66affSColin Finck 64c2c66affSColin Finck Urb->UrbControlTransfer.TransferFlags = USBD_SHORT_TRANSFER_OK | 65c2c66affSColin Finck USBD_TRANSFER_DIRECTION; 66c2c66affSColin Finck 67c2c66affSColin Finck if (SetupPacket->bmRequestType.Dir != BMREQUEST_DEVICE_TO_HOST) 68c2c66affSColin Finck { 69c2c66affSColin Finck Urb->UrbControlTransfer.TransferFlags &= ~USBD_TRANSFER_DIRECTION_IN; 70c2c66affSColin Finck } 71c2c66affSColin Finck 72c2c66affSColin Finck Status = STATUS_SUCCESS; 73c2c66affSColin Finck 74c2c66affSColin Finck if (Length) 75c2c66affSColin Finck { 76c2c66affSColin Finck Mdl = IoAllocateMdl(Buffer, Length, FALSE, FALSE, NULL); 77c2c66affSColin Finck 78c2c66affSColin Finck Urb->UrbControlTransfer.TransferBufferMDL = Mdl; 79c2c66affSColin Finck 80c2c66affSColin Finck if (Mdl) 81c2c66affSColin Finck { 82c2c66affSColin Finck Urb->UrbHeader.UsbdFlags |= USBD_FLAG_ALLOCATED_MDL; 83c2c66affSColin Finck MmBuildMdlForNonPagedPool(Mdl); 84c2c66affSColin Finck } 85c2c66affSColin Finck else 86c2c66affSColin Finck { 87c2c66affSColin Finck Status = USBPORT_USBDStatusToNtStatus(NULL, 88c2c66affSColin Finck USBD_STATUS_INSUFFICIENT_RESOURCES); 89c2c66affSColin Finck } 90c2c66affSColin Finck } 91c2c66affSColin Finck 92c2c66affSColin Finck if (NT_SUCCESS(Status)) 93c2c66affSColin Finck { 94c2c66affSColin Finck USBDStatus = USBPORT_AllocateTransfer(FdoDevice, 95c2c66affSColin Finck Urb, 96c2c66affSColin Finck NULL, 97c2c66affSColin Finck NULL, 98c2c66affSColin Finck &Event); 99c2c66affSColin Finck 100c2c66affSColin Finck if (USBD_SUCCESS(USBDStatus)) 101c2c66affSColin Finck { 102c2c66affSColin Finck InterlockedIncrement(&DeviceHandle->DeviceHandleLock); 103c2c66affSColin Finck 104c2c66affSColin Finck USBPORT_QueueTransferUrb(Urb); 105c2c66affSColin Finck 106c2c66affSColin Finck KeWaitForSingleObject(&Event, 107c2c66affSColin Finck Suspended, 108c2c66affSColin Finck KernelMode, 109c2c66affSColin Finck FALSE, 110c2c66affSColin Finck NULL); 111c2c66affSColin Finck 112c2c66affSColin Finck USBDStatus = Urb->UrbHeader.Status; 113c2c66affSColin Finck } 114c2c66affSColin Finck 115c2c66affSColin Finck Status = USBPORT_USBDStatusToNtStatus(Urb, USBDStatus); 116c2c66affSColin Finck 117c2c66affSColin Finck if (TransferedLen) 118c2c66affSColin Finck *TransferedLen = Urb->UrbControlTransfer.TransferBufferLength; 119c2c66affSColin Finck 120c2c66affSColin Finck if (pUSBDStatus) 121c2c66affSColin Finck *pUSBDStatus = USBDStatus; 122c2c66affSColin Finck } 123c2c66affSColin Finck 124c2c66affSColin Finck InterlockedDecrement(&DeviceHandle->DeviceHandleLock); 125c2c66affSColin Finck ExFreePoolWithTag(Urb, USB_PORT_TAG); 126c2c66affSColin Finck } 127c2c66affSColin Finck else 128c2c66affSColin Finck { 129c2c66affSColin Finck if (pUSBDStatus) 130c2c66affSColin Finck *pUSBDStatus = USBD_STATUS_INSUFFICIENT_RESOURCES; 131c2c66affSColin Finck 132c2c66affSColin Finck Status = USBPORT_USBDStatusToNtStatus(NULL, 133c2c66affSColin Finck USBD_STATUS_INSUFFICIENT_RESOURCES); 134c2c66affSColin Finck } 135c2c66affSColin Finck 136c2c66affSColin Finck DPRINT("USBPORT_SendSetupPacket: Status - %x\n", Status); 137c2c66affSColin Finck return Status; 138c2c66affSColin Finck } 139c2c66affSColin Finck 140c2c66affSColin Finck ULONG 141c2c66affSColin Finck NTAPI 142c2c66affSColin Finck USBPORT_GetInterfaceLength(IN PUSB_INTERFACE_DESCRIPTOR iDescriptor, 143c2c66affSColin Finck IN ULONG_PTR EndDescriptors) 144c2c66affSColin Finck { 145c2c66affSColin Finck SIZE_T Length; 146c2c66affSColin Finck PUSB_ENDPOINT_DESCRIPTOR Descriptor; 147c2c66affSColin Finck ULONG ix; 148c2c66affSColin Finck 149c2c66affSColin Finck DPRINT("USBPORT_GetInterfaceLength ... \n"); 150c2c66affSColin Finck 151c2c66affSColin Finck Length = iDescriptor->bLength; 152c2c66affSColin Finck Descriptor = (PUSB_ENDPOINT_DESCRIPTOR)((ULONG_PTR)iDescriptor + Length); 153c2c66affSColin Finck 154c2c66affSColin Finck if (iDescriptor->bNumEndpoints) 155c2c66affSColin Finck { 156c2c66affSColin Finck for (ix = 0; ix < iDescriptor->bNumEndpoints; ix++) 157c2c66affSColin Finck { 158c2c66affSColin Finck while ((Descriptor->bDescriptorType != USB_ENDPOINT_DESCRIPTOR_TYPE) && 159c2c66affSColin Finck (Descriptor->bLength > 0)) 160c2c66affSColin Finck { 161c2c66affSColin Finck Length += Descriptor->bLength; 162c2c66affSColin Finck Descriptor = (PUSB_ENDPOINT_DESCRIPTOR)((ULONG_PTR)Descriptor + 163c2c66affSColin Finck Descriptor->bLength); 164c2c66affSColin Finck } 165c2c66affSColin Finck 166c2c66affSColin Finck Length += Descriptor->bLength; 167c2c66affSColin Finck Descriptor = (PUSB_ENDPOINT_DESCRIPTOR)((ULONG_PTR)Descriptor + 168c2c66affSColin Finck Descriptor->bLength); 169c2c66affSColin Finck } 170c2c66affSColin Finck } 171c2c66affSColin Finck 172c2c66affSColin Finck while (((ULONG_PTR)Descriptor < EndDescriptors) && 173c2c66affSColin Finck (Descriptor->bDescriptorType != USB_INTERFACE_DESCRIPTOR_TYPE) && 174c2c66affSColin Finck (Descriptor->bLength > 0)) 175c2c66affSColin Finck { 176c2c66affSColin Finck Length += Descriptor->bLength; 177c2c66affSColin Finck Descriptor = (PUSB_ENDPOINT_DESCRIPTOR)((ULONG_PTR)Descriptor + 178c2c66affSColin Finck Descriptor->bLength); 179c2c66affSColin Finck } 180c2c66affSColin Finck 181c2c66affSColin Finck return Length; 182c2c66affSColin Finck } 183c2c66affSColin Finck 184c2c66affSColin Finck PUSB_INTERFACE_DESCRIPTOR 185c2c66affSColin Finck NTAPI 186c2c66affSColin Finck USBPORT_ParseConfigurationDescriptor(IN PUSB_CONFIGURATION_DESCRIPTOR ConfigDescriptor, 187c2c66affSColin Finck IN UCHAR InterfaceNumber, 188c2c66affSColin Finck IN UCHAR Alternate, 189c2c66affSColin Finck OUT PUCHAR OutAlternate) 190c2c66affSColin Finck { 191c2c66affSColin Finck PUSB_CONFIGURATION_DESCRIPTOR TmpDescriptor; 192c2c66affSColin Finck PUSB_INTERFACE_DESCRIPTOR iDescriptor; 193c2c66affSColin Finck PUSB_INTERFACE_DESCRIPTOR OutDescriptor = NULL; 194c2c66affSColin Finck ULONG_PTR Descriptor = (ULONG_PTR)ConfigDescriptor; 195c2c66affSColin Finck ULONG_PTR EndDescriptors; 196c2c66affSColin Finck ULONG ix; 197c2c66affSColin Finck 198c2c66affSColin Finck DPRINT("USBPORT_ParseConfigurationDescriptor ... \n"); 199c2c66affSColin Finck 200c2c66affSColin Finck if (OutAlternate) 201c2c66affSColin Finck *OutAlternate = 0; 202c2c66affSColin Finck 203c2c66affSColin Finck for (TmpDescriptor = (PUSB_CONFIGURATION_DESCRIPTOR)((ULONG_PTR)ConfigDescriptor + ConfigDescriptor->bLength); 204c2c66affSColin Finck TmpDescriptor->bDescriptorType == USB_CONFIGURATION_DESCRIPTOR_TYPE && TmpDescriptor->bDescriptorType > 0; 205c2c66affSColin Finck TmpDescriptor = (PUSB_CONFIGURATION_DESCRIPTOR)((ULONG_PTR)TmpDescriptor + TmpDescriptor->bLength)) 206c2c66affSColin Finck ; 207c2c66affSColin Finck 208c2c66affSColin Finck iDescriptor = (PUSB_INTERFACE_DESCRIPTOR)TmpDescriptor; 209c2c66affSColin Finck 210c2c66affSColin Finck EndDescriptors = (ULONG_PTR)ConfigDescriptor + 211c2c66affSColin Finck ConfigDescriptor->wTotalLength; 212c2c66affSColin Finck 213c2c66affSColin Finck while ((Descriptor < EndDescriptors) && 214c2c66affSColin Finck (iDescriptor->bInterfaceNumber != InterfaceNumber)) 215c2c66affSColin Finck { 216c2c66affSColin Finck Descriptor = (ULONG_PTR)iDescriptor + 217c2c66affSColin Finck USBPORT_GetInterfaceLength(iDescriptor, EndDescriptors); 218c2c66affSColin Finck 219c2c66affSColin Finck iDescriptor = (PUSB_INTERFACE_DESCRIPTOR)Descriptor; 220c2c66affSColin Finck } 221c2c66affSColin Finck 222c2c66affSColin Finck ix = 0; 223c2c66affSColin Finck 224c2c66affSColin Finck while (Descriptor < EndDescriptors && 225c2c66affSColin Finck iDescriptor->bInterfaceNumber == InterfaceNumber) 226c2c66affSColin Finck { 227c2c66affSColin Finck if (iDescriptor->bAlternateSetting == Alternate) 228c2c66affSColin Finck OutDescriptor = iDescriptor; 229c2c66affSColin Finck 230c2c66affSColin Finck Descriptor = (ULONG_PTR)iDescriptor + 231c2c66affSColin Finck USBPORT_GetInterfaceLength(iDescriptor, EndDescriptors); 232c2c66affSColin Finck 233c2c66affSColin Finck iDescriptor = (PUSB_INTERFACE_DESCRIPTOR)Descriptor; 234c2c66affSColin Finck 235c2c66affSColin Finck ++ix; 236c2c66affSColin Finck } 237c2c66affSColin Finck 238c2c66affSColin Finck if ((ix > 1) && OutAlternate) 239c2c66affSColin Finck *OutAlternate = 1; 240c2c66affSColin Finck 241c2c66affSColin Finck return OutDescriptor; 242c2c66affSColin Finck } 243c2c66affSColin Finck 244c2c66affSColin Finck USBD_STATUS 245c2c66affSColin Finck NTAPI 246c2c66affSColin Finck USBPORT_OpenInterface(IN PURB Urb, 247c2c66affSColin Finck IN PUSBPORT_DEVICE_HANDLE DeviceHandle, 248c2c66affSColin Finck IN PDEVICE_OBJECT FdoDevice, 249c2c66affSColin Finck IN PUSBPORT_CONFIGURATION_HANDLE ConfigHandle, 250c2c66affSColin Finck IN PUSBD_INTERFACE_INFORMATION InterfaceInfo, 251c2c66affSColin Finck IN OUT PUSBPORT_INTERFACE_HANDLE *iHandle, 252c2c66affSColin Finck IN BOOLEAN IsSetInterface) 253c2c66affSColin Finck { 254c2c66affSColin Finck PUSB_INTERFACE_DESCRIPTOR InterfaceDescriptor; 255c2c66affSColin Finck PUSBPORT_INTERFACE_HANDLE InterfaceHandle = NULL; 256c2c66affSColin Finck PUSBPORT_PIPE_HANDLE PipeHandle; 257c2c66affSColin Finck PUSB_ENDPOINT_DESCRIPTOR Descriptor; 258c2c66affSColin Finck PUSBD_PIPE_INFORMATION PipeInfo; 259c2c66affSColin Finck ULONG NumEndpoints; 260c2c66affSColin Finck SIZE_T Length; 261c2c66affSColin Finck SIZE_T HandleLength; 262c2c66affSColin Finck BOOLEAN IsAllocated = FALSE; 263c2c66affSColin Finck USHORT MaxPacketSize; 264c2c66affSColin Finck USHORT wMaxPacketSize; 265c2c66affSColin Finck ULONG ix; 266c2c66affSColin Finck USBD_STATUS USBDStatus = USBD_STATUS_SUCCESS; 267c2c66affSColin Finck NTSTATUS Status; 268c2c66affSColin Finck 269c2c66affSColin Finck DPRINT("USBPORT_OpenInterface: ...\n"); 270c2c66affSColin Finck 271c2c66affSColin Finck InterfaceDescriptor = USBPORT_ParseConfigurationDescriptor(ConfigHandle->ConfigurationDescriptor, 272c2c66affSColin Finck InterfaceInfo->InterfaceNumber, 273c2c66affSColin Finck InterfaceInfo->AlternateSetting, 274c2c66affSColin Finck &InterfaceInfo->AlternateSetting); 275c2c66affSColin Finck 276c2c66affSColin Finck NumEndpoints = InterfaceDescriptor->bNumEndpoints; 277c2c66affSColin Finck 278bf74391dSThomas Faber Length = FIELD_OFFSET(USBD_INTERFACE_INFORMATION, Pipes) + 279bf74391dSThomas Faber NumEndpoints * sizeof(USBD_PIPE_INFORMATION); 280c2c66affSColin Finck 281c2c66affSColin Finck if (InterfaceInfo->AlternateSetting && IsSetInterface) 282c2c66affSColin Finck { 283c2c66affSColin Finck DPRINT1("USBPORT_OpenInterface: InterfaceInfo->AlternateSetting && IsSetInterface !\n"); 284c2c66affSColin Finck } 285c2c66affSColin Finck 286c2c66affSColin Finck if (*iHandle) 287c2c66affSColin Finck { 288c2c66affSColin Finck InterfaceHandle = *iHandle; 289c2c66affSColin Finck } 290c2c66affSColin Finck else 291c2c66affSColin Finck { 292bf74391dSThomas Faber HandleLength = FIELD_OFFSET(USBPORT_INTERFACE_HANDLE, PipeHandle) + 293bf74391dSThomas Faber NumEndpoints * sizeof(USBPORT_PIPE_HANDLE); 294c2c66affSColin Finck 295c2c66affSColin Finck InterfaceHandle = ExAllocatePoolWithTag(NonPagedPool, 296c2c66affSColin Finck HandleLength, 297c2c66affSColin Finck USB_PORT_TAG); 298c2c66affSColin Finck 299c2c66affSColin Finck if (!InterfaceHandle) 300c2c66affSColin Finck { 301c2c66affSColin Finck USBDStatus = USBD_STATUS_INSUFFICIENT_RESOURCES; 302c2c66affSColin Finck goto Exit; 303c2c66affSColin Finck } 304c2c66affSColin Finck 305c2c66affSColin Finck RtlZeroMemory(InterfaceHandle, HandleLength); 306c2c66affSColin Finck 307c2c66affSColin Finck for (ix = 0; ix < NumEndpoints; ++ix) 308c2c66affSColin Finck { 309c2c66affSColin Finck PipeHandle = &InterfaceHandle->PipeHandle[ix]; 310c2c66affSColin Finck 311c2c66affSColin Finck PipeHandle->Flags = PIPE_HANDLE_FLAG_CLOSED; 312c2c66affSColin Finck PipeHandle->Endpoint = NULL; 313c2c66affSColin Finck } 314c2c66affSColin Finck 315c2c66affSColin Finck IsAllocated = TRUE; 316c2c66affSColin Finck } 317c2c66affSColin Finck 318c2c66affSColin Finck InterfaceHandle->AlternateSetting = InterfaceInfo->AlternateSetting; 319c2c66affSColin Finck 320c2c66affSColin Finck RtlCopyMemory(&InterfaceHandle->InterfaceDescriptor, 321c2c66affSColin Finck InterfaceDescriptor, 322c2c66affSColin Finck sizeof(USB_INTERFACE_DESCRIPTOR)); 323c2c66affSColin Finck 324c2c66affSColin Finck InterfaceInfo->Class = InterfaceDescriptor->bInterfaceClass; 325c2c66affSColin Finck InterfaceInfo->SubClass = InterfaceDescriptor->bInterfaceSubClass; 326c2c66affSColin Finck InterfaceInfo->Protocol = InterfaceDescriptor->bInterfaceProtocol; 327c2c66affSColin Finck InterfaceInfo->Reserved = 0; 328c2c66affSColin Finck InterfaceInfo->NumberOfPipes = InterfaceDescriptor->bNumEndpoints; 329c2c66affSColin Finck 330c2c66affSColin Finck Descriptor = (PUSB_ENDPOINT_DESCRIPTOR)((ULONG_PTR)InterfaceDescriptor + 331c2c66affSColin Finck InterfaceDescriptor->bLength); 332c2c66affSColin Finck 333c2c66affSColin Finck for (ix = 0; ix < NumEndpoints; ++ix) 334c2c66affSColin Finck { 335c2c66affSColin Finck PipeHandle = &InterfaceHandle->PipeHandle[ix]; 336c2c66affSColin Finck 337c2c66affSColin Finck while (Descriptor->bDescriptorType != USB_ENDPOINT_DESCRIPTOR_TYPE) 338c2c66affSColin Finck { 339c2c66affSColin Finck if (Descriptor->bLength == 0) 340c2c66affSColin Finck { 341c2c66affSColin Finck break; 342c2c66affSColin Finck } 343c2c66affSColin Finck else 344c2c66affSColin Finck { 345c2c66affSColin Finck Descriptor = (PUSB_ENDPOINT_DESCRIPTOR)((ULONG_PTR)Descriptor + 346c2c66affSColin Finck Descriptor->bLength); 347c2c66affSColin Finck } 348c2c66affSColin Finck } 349c2c66affSColin Finck 350c2c66affSColin Finck if (InterfaceInfo->Pipes[ix].PipeFlags & USBD_PF_CHANGE_MAX_PACKET) 351c2c66affSColin Finck { 352c2c66affSColin Finck Descriptor->wMaxPacketSize = InterfaceInfo->Pipes[ix].MaximumPacketSize; 353c2c66affSColin Finck } 354c2c66affSColin Finck 355c2c66affSColin Finck RtlCopyMemory(&PipeHandle->EndpointDescriptor, 356c2c66affSColin Finck Descriptor, 357c2c66affSColin Finck sizeof(USB_ENDPOINT_DESCRIPTOR)); 358c2c66affSColin Finck 359c2c66affSColin Finck PipeHandle->Flags = PIPE_HANDLE_FLAG_CLOSED; 360c2c66affSColin Finck PipeHandle->PipeFlags = InterfaceInfo->Pipes[ix].PipeFlags; 361c2c66affSColin Finck PipeHandle->Endpoint = NULL; 362c2c66affSColin Finck 363c2c66affSColin Finck wMaxPacketSize = Descriptor->wMaxPacketSize; 364c2c66affSColin Finck 365c2c66affSColin Finck /* USB 2.0 Specification, 5.9 High-Speed, High Bandwidth Endpoints */ 366c2c66affSColin Finck MaxPacketSize = (wMaxPacketSize & 0x7FF) * (((wMaxPacketSize >> 11) & 3) + 1); 367c2c66affSColin Finck 368c2c66affSColin Finck InterfaceInfo->Pipes[ix].EndpointAddress = Descriptor->bEndpointAddress; 369c2c66affSColin Finck InterfaceInfo->Pipes[ix].PipeType = Descriptor->bmAttributes & USB_ENDPOINT_TYPE_MASK; 370c2c66affSColin Finck InterfaceInfo->Pipes[ix].MaximumPacketSize = MaxPacketSize; 371c2c66affSColin Finck InterfaceInfo->Pipes[ix].PipeHandle = (USBD_PIPE_HANDLE)-1; 372c2c66affSColin Finck InterfaceInfo->Pipes[ix].Interval = Descriptor->bInterval; 373c2c66affSColin Finck 374c2c66affSColin Finck Descriptor = (PUSB_ENDPOINT_DESCRIPTOR)((ULONG_PTR)Descriptor + 375c2c66affSColin Finck Descriptor->bLength); 376c2c66affSColin Finck } 377c2c66affSColin Finck 378c2c66affSColin Finck if (USBD_SUCCESS(USBDStatus)) 379c2c66affSColin Finck { 380c2c66affSColin Finck for (ix = 0; ix < NumEndpoints; ++ix) 381c2c66affSColin Finck { 382c2c66affSColin Finck PipeInfo = &InterfaceInfo->Pipes[ix]; 383c2c66affSColin Finck PipeHandle = &InterfaceHandle->PipeHandle[ix]; 384c2c66affSColin Finck 385c2c66affSColin Finck Status = USBPORT_OpenPipe(FdoDevice, 386c2c66affSColin Finck DeviceHandle, 387c2c66affSColin Finck PipeHandle, 388c2c66affSColin Finck &USBDStatus); 389c2c66affSColin Finck 390c2c66affSColin Finck if (!NT_SUCCESS(Status)) 391c2c66affSColin Finck break; 392c2c66affSColin Finck 393c2c66affSColin Finck PipeInfo->PipeHandle = PipeHandle; 394c2c66affSColin Finck } 395c2c66affSColin Finck 396c2c66affSColin Finck if (NumEndpoints) 397c2c66affSColin Finck { 398c2c66affSColin Finck USBPORT_USBDStatusToNtStatus(Urb, USBDStatus); 399c2c66affSColin Finck } 400c2c66affSColin Finck } 401c2c66affSColin Finck 402c2c66affSColin Finck Exit: 403c2c66affSColin Finck 404c2c66affSColin Finck if (USBD_SUCCESS(USBDStatus)) 405c2c66affSColin Finck { 406c2c66affSColin Finck InterfaceInfo->InterfaceHandle = InterfaceHandle; 407c2c66affSColin Finck *iHandle = InterfaceHandle; 408c2c66affSColin Finck InterfaceInfo->Length = Length; 409c2c66affSColin Finck } 410c2c66affSColin Finck else 411c2c66affSColin Finck { 412c2c66affSColin Finck if (InterfaceHandle) 413c2c66affSColin Finck { 414c2c66affSColin Finck if (NumEndpoints) 415c2c66affSColin Finck { 416c2c66affSColin Finck DPRINT1("USBPORT_OpenInterface: USBDStatus - %lx\n", USBDStatus); 417c2c66affSColin Finck } 418c2c66affSColin Finck 419c2c66affSColin Finck if (IsAllocated) 420c2c66affSColin Finck ExFreePoolWithTag(InterfaceHandle, USB_PORT_TAG); 421c2c66affSColin Finck } 422c2c66affSColin Finck } 423c2c66affSColin Finck 424c2c66affSColin Finck return USBDStatus; 425c2c66affSColin Finck } 426c2c66affSColin Finck 427c2c66affSColin Finck VOID 428c2c66affSColin Finck NTAPI 429c2c66affSColin Finck USBPORT_CloseConfiguration(IN PUSBPORT_DEVICE_HANDLE DeviceHandle, 430c2c66affSColin Finck IN PDEVICE_OBJECT FdoDevice) 431c2c66affSColin Finck { 432c2c66affSColin Finck PUSBPORT_CONFIGURATION_HANDLE ConfigHandle; 433c2c66affSColin Finck PLIST_ENTRY iHandleList; 434c2c66affSColin Finck PUSBPORT_INTERFACE_HANDLE iHandle; 435c2c66affSColin Finck ULONG NumEndpoints; 436c2c66affSColin Finck PUSBPORT_PIPE_HANDLE PipeHandle; 437c2c66affSColin Finck 438c2c66affSColin Finck DPRINT("USBPORT_CloseConfiguration: ... \n"); 439c2c66affSColin Finck 440c2c66affSColin Finck ConfigHandle = DeviceHandle->ConfigHandle; 441c2c66affSColin Finck 442c2c66affSColin Finck if (ConfigHandle) 443c2c66affSColin Finck { 444c2c66affSColin Finck iHandleList = &ConfigHandle->InterfaceHandleList; 445c2c66affSColin Finck 446c2c66affSColin Finck while (!IsListEmpty(iHandleList)) 447c2c66affSColin Finck { 448c2c66affSColin Finck iHandle = CONTAINING_RECORD(iHandleList->Flink, 449c2c66affSColin Finck USBPORT_INTERFACE_HANDLE, 450c2c66affSColin Finck InterfaceLink); 451c2c66affSColin Finck 452c2c66affSColin Finck DPRINT("USBPORT_CloseConfiguration: iHandle - %p\n", iHandle); 453c2c66affSColin Finck 454c2c66affSColin Finck RemoveHeadList(iHandleList); 455c2c66affSColin Finck 456c2c66affSColin Finck NumEndpoints = iHandle->InterfaceDescriptor.bNumEndpoints; 457c2c66affSColin Finck 458c2c66affSColin Finck PipeHandle = &iHandle->PipeHandle[0]; 459c2c66affSColin Finck 460c2c66affSColin Finck while (NumEndpoints > 0) 461c2c66affSColin Finck { 462c2c66affSColin Finck USBPORT_ClosePipe(DeviceHandle, FdoDevice, PipeHandle); 463c2c66affSColin Finck PipeHandle += 1; 464c2c66affSColin Finck --NumEndpoints; 465c2c66affSColin Finck } 466c2c66affSColin Finck 467c2c66affSColin Finck ExFreePoolWithTag(iHandle, USB_PORT_TAG); 468c2c66affSColin Finck } 469c2c66affSColin Finck 470c2c66affSColin Finck ExFreePoolWithTag(ConfigHandle, USB_PORT_TAG); 471c2c66affSColin Finck DeviceHandle->ConfigHandle = NULL; 472c2c66affSColin Finck } 473c2c66affSColin Finck } 474c2c66affSColin Finck 475c2c66affSColin Finck NTSTATUS 476c2c66affSColin Finck NTAPI 477c2c66affSColin Finck USBPORT_InitInterfaceInfo(IN PUSBD_INTERFACE_INFORMATION InterfaceInfo, 478c2c66affSColin Finck IN PUSBPORT_CONFIGURATION_HANDLE ConfigHandle) 479c2c66affSColin Finck { 480c2c66affSColin Finck PUSB_INTERFACE_DESCRIPTOR Descriptor; 481c2c66affSColin Finck PUSBD_PIPE_INFORMATION Pipe; 482c2c66affSColin Finck USHORT Length; 483c2c66affSColin Finck ULONG PipeFlags; 484c2c66affSColin Finck ULONG NumberOfPipes; 485c2c66affSColin Finck USBD_STATUS USBDStatus = USBD_STATUS_SUCCESS; 486c2c66affSColin Finck 487c2c66affSColin Finck DPRINT("USBPORT_InitInterfaceInfo: InterfaceInfo - %p, ConfigHandle - %p\n", 488c2c66affSColin Finck InterfaceInfo, 489c2c66affSColin Finck ConfigHandle); 490c2c66affSColin Finck 491c2c66affSColin Finck Descriptor = USBPORT_ParseConfigurationDescriptor(ConfigHandle->ConfigurationDescriptor, 492c2c66affSColin Finck InterfaceInfo->InterfaceNumber, 493c2c66affSColin Finck InterfaceInfo->AlternateSetting, 494c2c66affSColin Finck &InterfaceInfo->AlternateSetting); 495c2c66affSColin Finck 496c2c66affSColin Finck Length = sizeof(USBD_INTERFACE_INFORMATION) + 497c2c66affSColin Finck sizeof(USBD_PIPE_INFORMATION); 498c2c66affSColin Finck 499c2c66affSColin Finck if (Descriptor) 500c2c66affSColin Finck { 501c2c66affSColin Finck NumberOfPipes = Descriptor->bNumEndpoints; 502c2c66affSColin Finck 503bf74391dSThomas Faber Length = FIELD_OFFSET(USBD_INTERFACE_INFORMATION, Pipes) + 504bf74391dSThomas Faber NumberOfPipes * sizeof(USBD_PIPE_INFORMATION); 505c2c66affSColin Finck 506c2c66affSColin Finck if (InterfaceInfo->Length >= Length) 507c2c66affSColin Finck { 508c2c66affSColin Finck InterfaceInfo->Class = 0; 509c2c66affSColin Finck InterfaceInfo->SubClass = 0; 510c2c66affSColin Finck InterfaceInfo->Protocol = 0; 511c2c66affSColin Finck InterfaceInfo->Reserved = 0; 512c2c66affSColin Finck InterfaceInfo->InterfaceHandle = 0; 513c2c66affSColin Finck InterfaceInfo->NumberOfPipes = NumberOfPipes; 514c2c66affSColin Finck 515c2c66affSColin Finck Pipe = InterfaceInfo->Pipes; 516c2c66affSColin Finck 517c2c66affSColin Finck while (NumberOfPipes > 0) 518c2c66affSColin Finck { 519c2c66affSColin Finck Pipe->EndpointAddress = 0; 520c2c66affSColin Finck Pipe->Interval = 0; 521c2c66affSColin Finck Pipe->PipeType = 0; 522c2c66affSColin Finck Pipe->PipeHandle = 0; 523c2c66affSColin Finck 524c2c66affSColin Finck PipeFlags = Pipe->PipeFlags; 525c2c66affSColin Finck 526c2c66affSColin Finck if (PipeFlags & ~USBD_PF_VALID_MASK) 527c2c66affSColin Finck USBDStatus = USBD_STATUS_INVALID_PIPE_FLAGS; 528c2c66affSColin Finck 529c2c66affSColin Finck if (!(PipeFlags & USBD_PF_CHANGE_MAX_PACKET)) 530c2c66affSColin Finck Pipe->MaximumPacketSize = 0; 531c2c66affSColin Finck 532c2c66affSColin Finck Pipe += 1; 533c2c66affSColin Finck --NumberOfPipes; 534c2c66affSColin Finck } 535c2c66affSColin Finck } 536c2c66affSColin Finck else 537c2c66affSColin Finck { 538c2c66affSColin Finck USBDStatus = USBD_STATUS_BUFFER_TOO_SMALL; 539c2c66affSColin Finck } 540c2c66affSColin Finck } 541c2c66affSColin Finck else 542c2c66affSColin Finck { 543c2c66affSColin Finck USBDStatus = USBD_STATUS_INTERFACE_NOT_FOUND; 544c2c66affSColin Finck } 545c2c66affSColin Finck 546c2c66affSColin Finck InterfaceInfo->Length = Length; 547c2c66affSColin Finck return USBDStatus; 548c2c66affSColin Finck } 549c2c66affSColin Finck 550c2c66affSColin Finck NTSTATUS 551c2c66affSColin Finck NTAPI 552c2c66affSColin Finck USBPORT_HandleSelectConfiguration(IN PDEVICE_OBJECT FdoDevice, 553c2c66affSColin Finck IN PIRP Irp, 554c2c66affSColin Finck IN PURB Urb) 555c2c66affSColin Finck { 556c2c66affSColin Finck PUSB_CONFIGURATION_DESCRIPTOR ConfigDescriptor; 557c2c66affSColin Finck PUSBPORT_DEVICE_HANDLE DeviceHandle; 558c2c66affSColin Finck PUSBPORT_CONFIGURATION_HANDLE ConfigHandle = NULL; 559c2c66affSColin Finck PUSBD_INTERFACE_INFORMATION InterfaceInfo; 560c2c66affSColin Finck PUSBPORT_INTERFACE_HANDLE InterfaceHandle; 561c2c66affSColin Finck ULONG iNumber; 562c2c66affSColin Finck ULONG ix; 563c2c66affSColin Finck USB_DEFAULT_PIPE_SETUP_PACKET SetupPacket; 564c2c66affSColin Finck NTSTATUS Status; 565c2c66affSColin Finck USBD_STATUS USBDStatus; 566c2c66affSColin Finck PUSBPORT_DEVICE_EXTENSION FdoExtension; 567c2c66affSColin Finck 568c2c66affSColin Finck DPRINT("USBPORT_HandleSelectConfiguration: ConfigDescriptor %p\n", 569c2c66affSColin Finck Urb->UrbSelectConfiguration.ConfigurationDescriptor); 570c2c66affSColin Finck 571c2c66affSColin Finck FdoExtension = FdoDevice->DeviceExtension; 572c2c66affSColin Finck 573c2c66affSColin Finck KeWaitForSingleObject(&FdoExtension->DeviceSemaphore, 574c2c66affSColin Finck Executive, 575c2c66affSColin Finck KernelMode, 576c2c66affSColin Finck FALSE, 577c2c66affSColin Finck NULL); 578c2c66affSColin Finck 579c2c66affSColin Finck DeviceHandle = Urb->UrbHeader.UsbdDeviceHandle; 580c2c66affSColin Finck ConfigDescriptor = Urb->UrbSelectConfiguration.ConfigurationDescriptor; 581c2c66affSColin Finck 582c2c66affSColin Finck if (!ConfigDescriptor) 583c2c66affSColin Finck { 584c2c66affSColin Finck DPRINT("USBPORT_HandleSelectConfiguration: ConfigDescriptor == NULL\n"); 585c2c66affSColin Finck 586c2c66affSColin Finck RtlZeroMemory(&SetupPacket, sizeof(USB_DEFAULT_PIPE_SETUP_PACKET)); 587c2c66affSColin Finck 588c2c66affSColin Finck SetupPacket.bmRequestType.B = 0; 589c2c66affSColin Finck SetupPacket.bRequest = USB_REQUEST_SET_CONFIGURATION; 590c2c66affSColin Finck SetupPacket.wValue.W = 0; 591c2c66affSColin Finck SetupPacket.wIndex.W = 0; 592c2c66affSColin Finck SetupPacket.wLength = 0; 593c2c66affSColin Finck 594c2c66affSColin Finck USBPORT_SendSetupPacket(DeviceHandle, 595c2c66affSColin Finck FdoDevice, 596c2c66affSColin Finck &SetupPacket, 597c2c66affSColin Finck NULL, 598c2c66affSColin Finck 0, 599c2c66affSColin Finck NULL, 600c2c66affSColin Finck NULL); 601c2c66affSColin Finck 602c2c66affSColin Finck Status = USBPORT_USBDStatusToNtStatus(Urb, USBD_STATUS_SUCCESS); 603c2c66affSColin Finck goto Exit; 604c2c66affSColin Finck } 605c2c66affSColin Finck 606c2c66affSColin Finck USBPORT_DumpingConfiguration(ConfigDescriptor); 607c2c66affSColin Finck 608c2c66affSColin Finck InterfaceInfo = &Urb->UrbSelectConfiguration.Interface; 609c2c66affSColin Finck 610c2c66affSColin Finck iNumber = 0; 611c2c66affSColin Finck 612c2c66affSColin Finck do 613c2c66affSColin Finck { 614c2c66affSColin Finck ++iNumber; 615c2c66affSColin Finck InterfaceInfo = (PUSBD_INTERFACE_INFORMATION) 616c2c66affSColin Finck ((ULONG_PTR)InterfaceInfo + 617c2c66affSColin Finck InterfaceInfo->Length); 618c2c66affSColin Finck } 619c2c66affSColin Finck while ((ULONG_PTR)InterfaceInfo < (ULONG_PTR)Urb + Urb->UrbHeader.Length); 620c2c66affSColin Finck 621c2c66affSColin Finck if ((iNumber <= 0) || (iNumber != ConfigDescriptor->bNumInterfaces)) 622c2c66affSColin Finck { 623c2c66affSColin Finck Status = USBPORT_USBDStatusToNtStatus(Urb, 624c2c66affSColin Finck USBD_STATUS_INVALID_CONFIGURATION_DESCRIPTOR); 625c2c66affSColin Finck goto Exit; 626c2c66affSColin Finck } 627c2c66affSColin Finck 628c2c66affSColin Finck ConfigHandle = ExAllocatePoolWithTag(NonPagedPool, 629c2c66affSColin Finck ConfigDescriptor->wTotalLength + sizeof(USBPORT_CONFIGURATION_HANDLE), 630c2c66affSColin Finck USB_PORT_TAG); 631c2c66affSColin Finck 632c2c66affSColin Finck if (!ConfigHandle) 633c2c66affSColin Finck { 634c2c66affSColin Finck Status = USBPORT_USBDStatusToNtStatus(Urb, 635c2c66affSColin Finck USBD_STATUS_INSUFFICIENT_RESOURCES); 636c2c66affSColin Finck goto Exit; 637c2c66affSColin Finck } 638c2c66affSColin Finck 639c2c66affSColin Finck RtlZeroMemory(ConfigHandle, 640c2c66affSColin Finck ConfigDescriptor->wTotalLength + sizeof(USBPORT_CONFIGURATION_HANDLE)); 641c2c66affSColin Finck 642c2c66affSColin Finck InitializeListHead(&ConfigHandle->InterfaceHandleList); 643c2c66affSColin Finck 644c2c66affSColin Finck ConfigHandle->ConfigurationDescriptor = (PUSB_CONFIGURATION_DESCRIPTOR)(ConfigHandle + 1); 645c2c66affSColin Finck 646c2c66affSColin Finck RtlCopyMemory(ConfigHandle->ConfigurationDescriptor, 647c2c66affSColin Finck ConfigDescriptor, 648c2c66affSColin Finck ConfigDescriptor->wTotalLength); 649c2c66affSColin Finck 650c2c66affSColin Finck RtlZeroMemory(&SetupPacket, sizeof(USB_DEFAULT_PIPE_SETUP_PACKET)); 651c2c66affSColin Finck 652c2c66affSColin Finck SetupPacket.bmRequestType.B = 0; 653c2c66affSColin Finck SetupPacket.bRequest = USB_REQUEST_SET_CONFIGURATION; 654c2c66affSColin Finck SetupPacket.wValue.W = ConfigDescriptor->bConfigurationValue; 655c2c66affSColin Finck SetupPacket.wIndex.W = 0; 656c2c66affSColin Finck SetupPacket.wLength = 0; 657c2c66affSColin Finck 658c2c66affSColin Finck USBPORT_SendSetupPacket(DeviceHandle, 659c2c66affSColin Finck FdoDevice, 660c2c66affSColin Finck &SetupPacket, 661c2c66affSColin Finck NULL, 662c2c66affSColin Finck 0, 663c2c66affSColin Finck NULL, 664c2c66affSColin Finck &USBDStatus); 665c2c66affSColin Finck 666c2c66affSColin Finck if (USBD_ERROR(USBDStatus)) 667c2c66affSColin Finck { 668c2c66affSColin Finck Status = USBPORT_USBDStatusToNtStatus(Urb, 669c2c66affSColin Finck USBD_STATUS_SET_CONFIG_FAILED); 670c2c66affSColin Finck goto Exit; 671c2c66affSColin Finck } 672c2c66affSColin Finck 673c2c66affSColin Finck if (iNumber <= 0) 674c2c66affSColin Finck { 675c2c66affSColin Finck Status = USBPORT_USBDStatusToNtStatus(Urb, 676c2c66affSColin Finck USBD_STATUS_SUCCESS); 677c2c66affSColin Finck 678c2c66affSColin Finck goto Exit; 679c2c66affSColin Finck } 680c2c66affSColin Finck 681c2c66affSColin Finck InterfaceInfo = &Urb->UrbSelectConfiguration.Interface; 682c2c66affSColin Finck 683c2c66affSColin Finck for (ix = 0; ix < iNumber; ++ix) 684c2c66affSColin Finck { 685c2c66affSColin Finck USBDStatus = USBPORT_InitInterfaceInfo(InterfaceInfo, 686c2c66affSColin Finck ConfigHandle); 687c2c66affSColin Finck 688c2c66affSColin Finck InterfaceHandle = NULL; 689c2c66affSColin Finck 690c2c66affSColin Finck if (USBD_SUCCESS(USBDStatus)) 691c2c66affSColin Finck { 692c2c66affSColin Finck USBDStatus = USBPORT_OpenInterface(Urb, 693c2c66affSColin Finck DeviceHandle, 694c2c66affSColin Finck FdoDevice, 695c2c66affSColin Finck ConfigHandle, 696c2c66affSColin Finck InterfaceInfo, 697c2c66affSColin Finck &InterfaceHandle, 698c2c66affSColin Finck TRUE); 699c2c66affSColin Finck } 700c2c66affSColin Finck 701c2c66affSColin Finck if (InterfaceHandle) 702c2c66affSColin Finck { 703c2c66affSColin Finck InsertTailList(&ConfigHandle->InterfaceHandleList, 704c2c66affSColin Finck &InterfaceHandle->InterfaceLink); 705c2c66affSColin Finck } 706c2c66affSColin Finck 707c2c66affSColin Finck if (USBD_ERROR(USBDStatus)) 708c2c66affSColin Finck break; 709c2c66affSColin Finck 710c2c66affSColin Finck InterfaceInfo = (PUSBD_INTERFACE_INFORMATION) 711c2c66affSColin Finck ((ULONG_PTR)InterfaceInfo + 712c2c66affSColin Finck InterfaceInfo->Length); 713c2c66affSColin Finck } 714c2c66affSColin Finck 715c2c66affSColin Finck if (ix >= iNumber) 716c2c66affSColin Finck { 717c2c66affSColin Finck Status = USBPORT_USBDStatusToNtStatus(Urb, 718c2c66affSColin Finck USBD_STATUS_SUCCESS); 719c2c66affSColin Finck } 720c2c66affSColin Finck else 721c2c66affSColin Finck { 722c2c66affSColin Finck Status = USBPORT_USBDStatusToNtStatus(Urb, USBDStatus); 723c2c66affSColin Finck } 724c2c66affSColin Finck 725c2c66affSColin Finck Exit: 726c2c66affSColin Finck 727c2c66affSColin Finck if (NT_SUCCESS(Status)) 728c2c66affSColin Finck { 729c2c66affSColin Finck Urb->UrbSelectConfiguration.ConfigurationHandle = ConfigHandle; 730c2c66affSColin Finck DeviceHandle->ConfigHandle = ConfigHandle; 731c2c66affSColin Finck } 732c2c66affSColin Finck else 733c2c66affSColin Finck { 734c2c66affSColin Finck DPRINT1("USBPORT_HandleSelectConfiguration: Status %x\n", Status); 735c2c66affSColin Finck } 736c2c66affSColin Finck 737c2c66affSColin Finck KeReleaseSemaphore(&FdoExtension->DeviceSemaphore, 738c2c66affSColin Finck LOW_REALTIME_PRIORITY, 739c2c66affSColin Finck 1, 740c2c66affSColin Finck FALSE); 741c2c66affSColin Finck 742c2c66affSColin Finck return Status; 743c2c66affSColin Finck } 744c2c66affSColin Finck 745c2c66affSColin Finck VOID 746c2c66affSColin Finck NTAPI 747c2c66affSColin Finck USBPORT_AddDeviceHandle(IN PDEVICE_OBJECT FdoDevice, 748c2c66affSColin Finck IN PUSBPORT_DEVICE_HANDLE DeviceHandle) 749c2c66affSColin Finck { 750c2c66affSColin Finck PUSBPORT_DEVICE_EXTENSION FdoExtension; 751c2c66affSColin Finck 752c2c66affSColin Finck DPRINT("USBPORT_AddDeviceHandle: ... \n"); 753c2c66affSColin Finck 754c2c66affSColin Finck FdoExtension = FdoDevice->DeviceExtension; 755c2c66affSColin Finck 756c2c66affSColin Finck InsertTailList(&FdoExtension->DeviceHandleList, 757c2c66affSColin Finck &DeviceHandle->DeviceHandleLink); 758c2c66affSColin Finck } 759c2c66affSColin Finck 760c2c66affSColin Finck VOID 761c2c66affSColin Finck NTAPI 762c2c66affSColin Finck USBPORT_RemoveDeviceHandle(IN PDEVICE_OBJECT FdoDevice, 763c2c66affSColin Finck IN PUSBPORT_DEVICE_HANDLE DeviceHandle) 764c2c66affSColin Finck { 765c2c66affSColin Finck PUSBPORT_DEVICE_EXTENSION FdoExtension; 766c2c66affSColin Finck KIRQL OldIrql; 767c2c66affSColin Finck 768c2c66affSColin Finck DPRINT("USBPORT_RemoveDeviceHandle \n"); 769c2c66affSColin Finck 770c2c66affSColin Finck FdoExtension = FdoDevice->DeviceExtension; 771c2c66affSColin Finck 772c2c66affSColin Finck KeAcquireSpinLock(&FdoExtension->DeviceHandleSpinLock, &OldIrql); 773c2c66affSColin Finck RemoveEntryList(&DeviceHandle->DeviceHandleLink); 774c2c66affSColin Finck KeReleaseSpinLock(&FdoExtension->DeviceHandleSpinLock, OldIrql); 775c2c66affSColin Finck } 776c2c66affSColin Finck 777c2c66affSColin Finck BOOLEAN 778c2c66affSColin Finck NTAPI 779c2c66affSColin Finck USBPORT_ValidateDeviceHandle(IN PDEVICE_OBJECT FdoDevice, 780c2c66affSColin Finck IN PUSBPORT_DEVICE_HANDLE DeviceHandle) 781c2c66affSColin Finck { 782c2c66affSColin Finck PUSBPORT_DEVICE_EXTENSION FdoExtension; 783c2c66affSColin Finck KIRQL OldIrql; 784c2c66affSColin Finck PLIST_ENTRY HandleList; 785c2c66affSColin Finck PUSBPORT_DEVICE_HANDLE CurrentHandle; 786c2c66affSColin Finck BOOLEAN Result = FALSE; 787c2c66affSColin Finck 788c2c66affSColin Finck //DPRINT("USBPORT_ValidateDeviceHandle: DeviceHandle - %p\n", DeviceHandle \n"); 789c2c66affSColin Finck 790c2c66affSColin Finck FdoExtension = FdoDevice->DeviceExtension; 791c2c66affSColin Finck 792c2c66affSColin Finck KeAcquireSpinLock(&FdoExtension->DeviceHandleSpinLock, &OldIrql); 793c2c66affSColin Finck if (DeviceHandle) 794c2c66affSColin Finck { 795c2c66affSColin Finck HandleList = FdoExtension->DeviceHandleList.Flink; 796c2c66affSColin Finck 797c2c66affSColin Finck while (HandleList != &FdoExtension->DeviceHandleList) 798c2c66affSColin Finck { 799c2c66affSColin Finck CurrentHandle = CONTAINING_RECORD(HandleList, 800c2c66affSColin Finck USBPORT_DEVICE_HANDLE, 801c2c66affSColin Finck DeviceHandleLink); 802c2c66affSColin Finck 803c2c66affSColin Finck if (CurrentHandle == DeviceHandle) 804c2c66affSColin Finck { 805c2c66affSColin Finck Result = TRUE; 806c2c66affSColin Finck break; 807c2c66affSColin Finck } 808c2c66affSColin Finck 809c2c66affSColin Finck HandleList = HandleList->Flink; 810c2c66affSColin Finck } 811c2c66affSColin Finck } 812c2c66affSColin Finck KeReleaseSpinLock(&FdoExtension->DeviceHandleSpinLock, OldIrql); 813c2c66affSColin Finck 814c2c66affSColin Finck return Result; 815c2c66affSColin Finck } 816c2c66affSColin Finck 817c2c66affSColin Finck BOOLEAN 818c2c66affSColin Finck NTAPI 819c2c66affSColin Finck USBPORT_DeviceHasTransfers(IN PDEVICE_OBJECT FdoDevice, 820c2c66affSColin Finck IN PUSBPORT_DEVICE_HANDLE DeviceHandle) 821c2c66affSColin Finck { 822c2c66affSColin Finck PLIST_ENTRY PipeHandleList; 823c2c66affSColin Finck PUSBPORT_PIPE_HANDLE PipeHandle; 824c2c66affSColin Finck 825c2c66affSColin Finck DPRINT("USBPORT_DeviceHasTransfers: ... \n"); 826c2c66affSColin Finck 827c2c66affSColin Finck PipeHandleList = DeviceHandle->PipeHandleList.Flink; 828c2c66affSColin Finck 829c2c66affSColin Finck while (PipeHandleList != &DeviceHandle->PipeHandleList) 830c2c66affSColin Finck { 831c2c66affSColin Finck PipeHandle = CONTAINING_RECORD(PipeHandleList, 832c2c66affSColin Finck USBPORT_PIPE_HANDLE, 833c2c66affSColin Finck PipeLink); 834c2c66affSColin Finck 835c2c66affSColin Finck PipeHandleList = PipeHandleList->Flink; 836c2c66affSColin Finck 837c2c66affSColin Finck if (!(PipeHandle->Flags & PIPE_HANDLE_FLAG_NULL_PACKET_SIZE) && 838c2c66affSColin Finck USBPORT_EndpointHasQueuedTransfers(FdoDevice, PipeHandle->Endpoint, NULL)) 839c2c66affSColin Finck { 840c2c66affSColin Finck return TRUE; 841c2c66affSColin Finck } 842c2c66affSColin Finck } 843c2c66affSColin Finck 844c2c66affSColin Finck return FALSE; 845c2c66affSColin Finck } 846c2c66affSColin Finck 847c2c66affSColin Finck VOID 848c2c66affSColin Finck NTAPI 849c2c66affSColin Finck USBPORT_AbortTransfers(IN PDEVICE_OBJECT FdoDevice, 850c2c66affSColin Finck IN PUSBPORT_DEVICE_HANDLE DeviceHandle) 851c2c66affSColin Finck { 852c2c66affSColin Finck PLIST_ENTRY HandleList; 853c2c66affSColin Finck PUSBPORT_PIPE_HANDLE PipeHandle; 854c2c66affSColin Finck BOOLEAN Result; 855c2c66affSColin Finck 856c2c66affSColin Finck DPRINT("USBPORT_AbortAllTransfers: ... \n"); 857c2c66affSColin Finck 858c2c66affSColin Finck HandleList = DeviceHandle->PipeHandleList.Flink; 859c2c66affSColin Finck 860c2c66affSColin Finck while (HandleList != &DeviceHandle->PipeHandleList) 861c2c66affSColin Finck { 862c2c66affSColin Finck PipeHandle = CONTAINING_RECORD(HandleList, 863c2c66affSColin Finck USBPORT_PIPE_HANDLE, 864c2c66affSColin Finck PipeLink); 865c2c66affSColin Finck 866c2c66affSColin Finck HandleList = HandleList->Flink; 867c2c66affSColin Finck 868c2c66affSColin Finck if (!(PipeHandle->Flags & DEVICE_HANDLE_FLAG_ROOTHUB)) 869c2c66affSColin Finck { 870c2c66affSColin Finck PipeHandle->Endpoint->Flags |= ENDPOINT_FLAG_ABORTING; 871c2c66affSColin Finck 872c2c66affSColin Finck USBPORT_AbortEndpoint(FdoDevice, PipeHandle->Endpoint, NULL); 873c2c66affSColin Finck USBPORT_FlushMapTransfers(FdoDevice); 874c2c66affSColin Finck } 875c2c66affSColin Finck } 876c2c66affSColin Finck 877c2c66affSColin Finck while (TRUE) 878c2c66affSColin Finck { 879c2c66affSColin Finck Result = USBPORT_DeviceHasTransfers(FdoDevice, DeviceHandle); 880c2c66affSColin Finck 881c2c66affSColin Finck if (!Result) 882c2c66affSColin Finck break; 883c2c66affSColin Finck 884c2c66affSColin Finck USBPORT_Wait(FdoDevice, 100); 885c2c66affSColin Finck } 886c2c66affSColin Finck } 887c2c66affSColin Finck 8880f8425edSVadim Galyant PUSB2_TT_EXTENSION 8890f8425edSVadim Galyant NTAPI 8900f8425edSVadim Galyant USBPORT_GetTt(IN PDEVICE_OBJECT FdoDevice, 8910f8425edSVadim Galyant IN PUSBPORT_DEVICE_HANDLE HubDeviceHandle, 8920f8425edSVadim Galyant OUT PUSHORT OutPort, 8930f8425edSVadim Galyant OUT PUSBPORT_DEVICE_HANDLE * OutHubDeviceHandle) 8940f8425edSVadim Galyant { 8950f8425edSVadim Galyant PUSBPORT_DEVICE_HANDLE DeviceHandle = HubDeviceHandle; 8960f8425edSVadim Galyant ULONG TtCount; 8970f8425edSVadim Galyant PLIST_ENTRY Entry; 8980f8425edSVadim Galyant PUSB2_TT_EXTENSION TtExtension = NULL; 8990f8425edSVadim Galyant 9000f8425edSVadim Galyant DPRINT("USBPORT_GetTt: HubDeviceHandle - %p\n", HubDeviceHandle); 9010f8425edSVadim Galyant 9020f8425edSVadim Galyant *OutHubDeviceHandle = NULL; 9030f8425edSVadim Galyant 9040f8425edSVadim Galyant while (DeviceHandle->DeviceSpeed != UsbHighSpeed) 9050f8425edSVadim Galyant { 9060f8425edSVadim Galyant DPRINT("USBPORT_GetTt: DeviceHandle - %p, DeviceHandle->PortNumber - %X\n", 9070f8425edSVadim Galyant DeviceHandle, 9080f8425edSVadim Galyant DeviceHandle->PortNumber); 9090f8425edSVadim Galyant 9100f8425edSVadim Galyant *OutPort = DeviceHandle->PortNumber; 9110f8425edSVadim Galyant 9120f8425edSVadim Galyant DeviceHandle = DeviceHandle->HubDeviceHandle; 9130f8425edSVadim Galyant 9140f8425edSVadim Galyant if (!DeviceHandle) 9150f8425edSVadim Galyant return NULL; 9160f8425edSVadim Galyant } 9170f8425edSVadim Galyant 9180f8425edSVadim Galyant TtCount = DeviceHandle->TtCount; 9190f8425edSVadim Galyant 9200f8425edSVadim Galyant if (!TtCount) 9210f8425edSVadim Galyant return NULL; 9220f8425edSVadim Galyant 9230f8425edSVadim Galyant if (IsListEmpty(&DeviceHandle->TtList)) 9240f8425edSVadim Galyant return NULL; 9250f8425edSVadim Galyant 9260f8425edSVadim Galyant Entry = DeviceHandle->TtList.Flink; 9270f8425edSVadim Galyant 9280f8425edSVadim Galyant if (TtCount > 1) 9290f8425edSVadim Galyant { 9300f8425edSVadim Galyant while (Entry != &DeviceHandle->TtList) 9310f8425edSVadim Galyant { 9320f8425edSVadim Galyant ASSERT(Entry != NULL); 9330f8425edSVadim Galyant 9340f8425edSVadim Galyant TtExtension = CONTAINING_RECORD(Entry, 9350f8425edSVadim Galyant USB2_TT_EXTENSION, 9360f8425edSVadim Galyant Link); 9370f8425edSVadim Galyant 9380f8425edSVadim Galyant if (TtExtension->TtNumber == *OutPort) 9390f8425edSVadim Galyant break; 9400f8425edSVadim Galyant 9410f8425edSVadim Galyant Entry = Entry->Flink; 9420f8425edSVadim Galyant 9430f8425edSVadim Galyant TtExtension = NULL; 9440f8425edSVadim Galyant } 9450f8425edSVadim Galyant } 9460f8425edSVadim Galyant else 9470f8425edSVadim Galyant { 9480f8425edSVadim Galyant TtExtension = CONTAINING_RECORD(Entry, 9490f8425edSVadim Galyant USB2_TT_EXTENSION, 9500f8425edSVadim Galyant Link); 9510f8425edSVadim Galyant } 9520f8425edSVadim Galyant 9530f8425edSVadim Galyant *OutHubDeviceHandle = DeviceHandle; 9540f8425edSVadim Galyant 9550f8425edSVadim Galyant return TtExtension; 9560f8425edSVadim Galyant } 9570f8425edSVadim Galyant 958c2c66affSColin Finck NTSTATUS 959c2c66affSColin Finck NTAPI 960c2c66affSColin Finck USBPORT_CreateDevice(IN OUT PUSB_DEVICE_HANDLE *pUsbdDeviceHandle, 961c2c66affSColin Finck IN PDEVICE_OBJECT FdoDevice, 962c2c66affSColin Finck IN PUSBPORT_DEVICE_HANDLE HubDeviceHandle, 963c2c66affSColin Finck IN USHORT PortStatus, 964c2c66affSColin Finck IN USHORT Port) 965c2c66affSColin Finck { 966c92f5911SVadim Galyant PUSBPORT_DEVICE_HANDLE TtDeviceHandle = NULL; 967c92f5911SVadim Galyant PUSB2_TT_EXTENSION TtExtension = NULL; 968c92f5911SVadim Galyant USHORT port; 969c2c66affSColin Finck PUSBPORT_DEVICE_HANDLE DeviceHandle; 970c2c66affSColin Finck PUSBPORT_PIPE_HANDLE PipeHandle; 971c2c66affSColin Finck BOOL IsOpenedPipe; 972c2c66affSColin Finck PVOID DeviceDescriptor; 973c2c66affSColin Finck USB_DEFAULT_PIPE_SETUP_PACKET SetupPacket; 974c2c66affSColin Finck SIZE_T TransferedLen; 975c2c66affSColin Finck SIZE_T DescriptorMinSize; 976c2c66affSColin Finck UCHAR MaxPacketSize; 977c2c66affSColin Finck PUSBPORT_DEVICE_EXTENSION FdoExtension; 978c92f5911SVadim Galyant PUSBPORT_REGISTRATION_PACKET Packet; 979c2c66affSColin Finck NTSTATUS Status; 980c2c66affSColin Finck 981c2c66affSColin Finck DPRINT("USBPORT_CreateDevice: PortStatus - %p, Port - %x\n", 982c2c66affSColin Finck PortStatus, 983c2c66affSColin Finck Port); 984c2c66affSColin Finck 985c2c66affSColin Finck FdoExtension = FdoDevice->DeviceExtension; 986c92f5911SVadim Galyant Packet = &FdoExtension->MiniPortInterface->Packet; 987c2c66affSColin Finck 988c2c66affSColin Finck KeWaitForSingleObject(&FdoExtension->DeviceSemaphore, 989c2c66affSColin Finck Executive, 990c2c66affSColin Finck KernelMode, 991c2c66affSColin Finck FALSE, 992c2c66affSColin Finck NULL); 993c2c66affSColin Finck 994c2c66affSColin Finck if (!USBPORT_ValidateDeviceHandle(FdoDevice, HubDeviceHandle)) 995c2c66affSColin Finck { 996c2c66affSColin Finck KeReleaseSemaphore(&FdoExtension->DeviceSemaphore, 997c2c66affSColin Finck LOW_REALTIME_PRIORITY, 998c2c66affSColin Finck 1, 999c2c66affSColin Finck FALSE); 1000c2c66affSColin Finck 1001c2c66affSColin Finck DPRINT1("USBPORT_CreateDevice: Not valid hub DeviceHandle\n"); 1002c2c66affSColin Finck return STATUS_DEVICE_NOT_CONNECTED; 1003c2c66affSColin Finck } 1004c2c66affSColin Finck 1005c92f5911SVadim Galyant port = Port; 1006c92f5911SVadim Galyant 1007c92f5911SVadim Galyant if (Packet->MiniPortFlags & USB_MINIPORT_FLAGS_USB2 && 1008c2c66affSColin Finck !(PortStatus & USB_PORT_STATUS_HIGH_SPEED)) 1009c2c66affSColin Finck { 1010c92f5911SVadim Galyant DPRINT1("USBPORT_CreateDevice: USB1 device connected to USB2 port\n"); 1011c92f5911SVadim Galyant 1012c92f5911SVadim Galyant TtExtension = USBPORT_GetTt(FdoDevice, 1013c92f5911SVadim Galyant HubDeviceHandle, 1014c92f5911SVadim Galyant &port, 1015c92f5911SVadim Galyant &TtDeviceHandle); 1016c92f5911SVadim Galyant 1017c92f5911SVadim Galyant DPRINT("USBPORT_CreateDevice: TtDeviceHandle - %p, port - %x\n", 1018c92f5911SVadim Galyant TtDeviceHandle, 1019c92f5911SVadim Galyant port); 1020c2c66affSColin Finck } 1021c2c66affSColin Finck 1022c2c66affSColin Finck KeReleaseSemaphore(&FdoExtension->DeviceSemaphore, 1023c2c66affSColin Finck LOW_REALTIME_PRIORITY, 1024c2c66affSColin Finck 1, 1025c2c66affSColin Finck FALSE); 1026c2c66affSColin Finck 1027c2c66affSColin Finck DeviceHandle = ExAllocatePoolWithTag(NonPagedPool, 1028c2c66affSColin Finck sizeof(USBPORT_DEVICE_HANDLE), 1029c2c66affSColin Finck USB_PORT_TAG); 1030c2c66affSColin Finck 1031c2c66affSColin Finck if (!DeviceHandle) 1032c2c66affSColin Finck { 1033c2c66affSColin Finck DPRINT1("USBPORT_CreateDevice: Not allocated DeviceHandle\n"); 1034c2c66affSColin Finck return STATUS_INSUFFICIENT_RESOURCES; 1035c2c66affSColin Finck } 1036c2c66affSColin Finck 1037c2c66affSColin Finck RtlZeroMemory(DeviceHandle, sizeof(USBPORT_DEVICE_HANDLE)); 1038c2c66affSColin Finck 1039c2c66affSColin Finck *pUsbdDeviceHandle = NULL; 1040c2c66affSColin Finck 1041c92f5911SVadim Galyant DeviceHandle->TtExtension = TtExtension; 1042c2c66affSColin Finck DeviceHandle->PortNumber = Port; 1043c2c66affSColin Finck DeviceHandle->HubDeviceHandle = HubDeviceHandle; 1044c2c66affSColin Finck 1045c2c66affSColin Finck if (PortStatus & USB_PORT_STATUS_LOW_SPEED) 1046c2c66affSColin Finck { 1047c2c66affSColin Finck DeviceHandle->DeviceSpeed = UsbLowSpeed; 1048c2c66affSColin Finck } 1049c2c66affSColin Finck else if (PortStatus & USB_PORT_STATUS_HIGH_SPEED) 1050c2c66affSColin Finck { 1051c2c66affSColin Finck DeviceHandle->DeviceSpeed = UsbHighSpeed; 1052c2c66affSColin Finck } 1053c2c66affSColin Finck else 1054c2c66affSColin Finck { 1055c2c66affSColin Finck DeviceHandle->DeviceSpeed = UsbFullSpeed; 1056c2c66affSColin Finck } 1057c2c66affSColin Finck 1058c2c66affSColin Finck KeWaitForSingleObject(&FdoExtension->DeviceSemaphore, 1059c2c66affSColin Finck Executive, 1060c2c66affSColin Finck KernelMode, 1061c2c66affSColin Finck FALSE, 1062c2c66affSColin Finck NULL); 1063c2c66affSColin Finck 1064c2c66affSColin Finck PipeHandle = &DeviceHandle->PipeHandle; 1065c2c66affSColin Finck 1066c2c66affSColin Finck PipeHandle->Flags = PIPE_HANDLE_FLAG_CLOSED; 1067c2c66affSColin Finck 1068c2c66affSColin Finck PipeHandle->EndpointDescriptor.bLength = sizeof(PipeHandle->EndpointDescriptor); 1069c2c66affSColin Finck PipeHandle->EndpointDescriptor.bDescriptorType = USB_ENDPOINT_DESCRIPTOR_TYPE; 1070c2c66affSColin Finck 1071c2c66affSColin Finck if (DeviceHandle->DeviceSpeed == UsbLowSpeed) 1072c2c66affSColin Finck { 1073c2c66affSColin Finck PipeHandle->EndpointDescriptor.wMaxPacketSize = 8; 1074c2c66affSColin Finck } 1075c2c66affSColin Finck else 1076c2c66affSColin Finck { 1077c2c66affSColin Finck PipeHandle->EndpointDescriptor.wMaxPacketSize = USB_DEFAULT_MAX_PACKET; 1078c2c66affSColin Finck } 1079c2c66affSColin Finck 1080c2c66affSColin Finck InitializeListHead(&DeviceHandle->PipeHandleList); 1081c92f5911SVadim Galyant InitializeListHead(&DeviceHandle->TtList); 1082c2c66affSColin Finck 1083c2c66affSColin Finck Status = USBPORT_OpenPipe(FdoDevice, 1084c2c66affSColin Finck DeviceHandle, 1085c2c66affSColin Finck PipeHandle, 1086c2c66affSColin Finck NULL); 1087c2c66affSColin Finck 1088c2c66affSColin Finck IsOpenedPipe = NT_SUCCESS(Status); 1089c2c66affSColin Finck 1090c2c66affSColin Finck if (NT_ERROR(Status)) 1091c2c66affSColin Finck { 1092c2c66affSColin Finck DPRINT1("USBPORT_CreateDevice: USBPORT_OpenPipe return - %lx\n", Status); 1093c2c66affSColin Finck 1094c2c66affSColin Finck KeReleaseSemaphore(&FdoExtension->DeviceSemaphore, 1095c2c66affSColin Finck LOW_REALTIME_PRIORITY, 1096c2c66affSColin Finck 1, 1097c2c66affSColin Finck FALSE); 1098c2c66affSColin Finck 1099c2c66affSColin Finck ExFreePoolWithTag(DeviceHandle, USB_PORT_TAG); 1100c2c66affSColin Finck 1101c2c66affSColin Finck return Status; 1102c2c66affSColin Finck } 1103c2c66affSColin Finck 1104c2c66affSColin Finck DeviceDescriptor = ExAllocatePoolWithTag(NonPagedPool, 1105c2c66affSColin Finck USB_DEFAULT_MAX_PACKET, 1106c2c66affSColin Finck USB_PORT_TAG); 1107c2c66affSColin Finck 1108c2c66affSColin Finck if (!DeviceDescriptor) 1109c2c66affSColin Finck { 1110c2c66affSColin Finck DPRINT1("USBPORT_CreateDevice: Not allocated DeviceDescriptor\n"); 1111c2c66affSColin Finck goto ErrorExit; 1112c2c66affSColin Finck } 1113c2c66affSColin Finck 1114c2c66affSColin Finck RtlZeroMemory(DeviceDescriptor, USB_DEFAULT_MAX_PACKET); 1115c2c66affSColin Finck RtlZeroMemory(&SetupPacket, sizeof(USB_DEFAULT_PIPE_SETUP_PACKET)); 1116c2c66affSColin Finck 1117c2c66affSColin Finck SetupPacket.bmRequestType.Dir = BMREQUEST_DEVICE_TO_HOST; 1118c2c66affSColin Finck SetupPacket.bRequest = USB_REQUEST_GET_DESCRIPTOR; 1119c2c66affSColin Finck SetupPacket.wValue.HiByte = USB_DEVICE_DESCRIPTOR_TYPE; 1120c2c66affSColin Finck SetupPacket.wLength = USB_DEFAULT_MAX_PACKET; 1121c2c66affSColin Finck 1122c2c66affSColin Finck TransferedLen = 0; 1123c2c66affSColin Finck 1124c2c66affSColin Finck Status = USBPORT_SendSetupPacket(DeviceHandle, 1125c2c66affSColin Finck FdoDevice, 1126c2c66affSColin Finck &SetupPacket, 1127c2c66affSColin Finck DeviceDescriptor, 1128c2c66affSColin Finck USB_DEFAULT_MAX_PACKET, 1129c2c66affSColin Finck &TransferedLen, 1130c2c66affSColin Finck NULL); 1131c2c66affSColin Finck 1132c2c66affSColin Finck RtlCopyMemory(&DeviceHandle->DeviceDescriptor, 1133c2c66affSColin Finck DeviceDescriptor, 1134c2c66affSColin Finck sizeof(USB_DEVICE_DESCRIPTOR)); 1135c2c66affSColin Finck 1136c2c66affSColin Finck ExFreePoolWithTag(DeviceDescriptor, USB_PORT_TAG); 1137c2c66affSColin Finck 1138c2c66affSColin Finck DescriptorMinSize = RTL_SIZEOF_THROUGH_FIELD(USB_DEVICE_DESCRIPTOR, 1139c2c66affSColin Finck bMaxPacketSize0); 1140c2c66affSColin Finck 1141c2c66affSColin Finck if ((TransferedLen == DescriptorMinSize) && !NT_SUCCESS(Status)) 1142c2c66affSColin Finck { 1143c2c66affSColin Finck Status = STATUS_SUCCESS; 1144c2c66affSColin Finck } 1145c2c66affSColin Finck 1146c2c66affSColin Finck if (NT_SUCCESS(Status) && (TransferedLen >= DescriptorMinSize)) 1147c2c66affSColin Finck { 1148c2c66affSColin Finck if ((DeviceHandle->DeviceDescriptor.bLength >= sizeof(USB_DEVICE_DESCRIPTOR)) && 1149c2c66affSColin Finck (DeviceHandle->DeviceDescriptor.bDescriptorType == USB_DEVICE_DESCRIPTOR_TYPE)) 1150c2c66affSColin Finck { 1151c2c66affSColin Finck MaxPacketSize = DeviceHandle->DeviceDescriptor.bMaxPacketSize0; 1152c2c66affSColin Finck 1153c2c66affSColin Finck if (MaxPacketSize == 8 || 1154c2c66affSColin Finck MaxPacketSize == 16 || 1155c2c66affSColin Finck MaxPacketSize == 32 || 1156c2c66affSColin Finck MaxPacketSize == 64) 1157c2c66affSColin Finck { 1158c2c66affSColin Finck USBPORT_AddDeviceHandle(FdoDevice, DeviceHandle); 1159c2c66affSColin Finck 1160c2c66affSColin Finck *pUsbdDeviceHandle = DeviceHandle; 1161c2c66affSColin Finck 1162c2c66affSColin Finck KeReleaseSemaphore(&FdoExtension->DeviceSemaphore, 1163c2c66affSColin Finck LOW_REALTIME_PRIORITY, 1164c2c66affSColin Finck 1, 1165c2c66affSColin Finck FALSE); 1166c2c66affSColin Finck 1167c2c66affSColin Finck return Status; 1168c2c66affSColin Finck } 1169c2c66affSColin Finck } 1170c2c66affSColin Finck } 1171c2c66affSColin Finck 1172c2c66affSColin Finck DPRINT1("USBPORT_CreateDevice: ERROR!!! TransferedLen - %x, Status - %lx\n", 1173c2c66affSColin Finck TransferedLen, 1174c2c66affSColin Finck Status); 1175c2c66affSColin Finck 1176c2c66affSColin Finck ErrorExit: 1177c2c66affSColin Finck 1178c92f5911SVadim Galyant if (TtExtension && TtDeviceHandle) 1179c92f5911SVadim Galyant { 1180c92f5911SVadim Galyant SetupPacket.bmRequestType.Recipient = BMREQUEST_TO_OTHER; 1181c92f5911SVadim Galyant SetupPacket.bmRequestType.Reserved = 0; 1182c92f5911SVadim Galyant SetupPacket.bmRequestType.Type = BMREQUEST_CLASS; 1183c92f5911SVadim Galyant SetupPacket.bmRequestType.Dir = BMREQUEST_HOST_TO_DEVICE; 1184c92f5911SVadim Galyant 1185c92f5911SVadim Galyant /* Table 11-15. Hub Class Requests */ 1186c92f5911SVadim Galyant if (TtDeviceHandle == HubDeviceHandle) 1187c92f5911SVadim Galyant { 1188c92f5911SVadim Galyant SetupPacket.bRequest = USB_REQUEST_RESET_TT; 1189c92f5911SVadim Galyant } 1190c92f5911SVadim Galyant else 1191c92f5911SVadim Galyant { 1192c92f5911SVadim Galyant SetupPacket.bRequest = USB_REQUEST_CLEAR_TT_BUFFER; 1193c92f5911SVadim Galyant } 1194c92f5911SVadim Galyant 1195c92f5911SVadim Galyant SetupPacket.wValue.LowByte = 0; 1196c92f5911SVadim Galyant SetupPacket.wValue.HiByte = 0; 1197c92f5911SVadim Galyant SetupPacket.wIndex.W = port; 1198c92f5911SVadim Galyant SetupPacket.wLength = 0; 1199c92f5911SVadim Galyant 1200c92f5911SVadim Galyant USBPORT_SendSetupPacket(TtDeviceHandle, 1201c92f5911SVadim Galyant FdoDevice, 1202c92f5911SVadim Galyant &SetupPacket, 1203c92f5911SVadim Galyant NULL, 1204c92f5911SVadim Galyant 0, 1205c92f5911SVadim Galyant NULL, 1206c92f5911SVadim Galyant NULL); 1207c92f5911SVadim Galyant } 1208c2c66affSColin Finck 1209c2c66affSColin Finck Status = STATUS_DEVICE_DATA_ERROR; 1210c2c66affSColin Finck 1211c2c66affSColin Finck if (IsOpenedPipe) 1212c2c66affSColin Finck { 1213c2c66affSColin Finck USBPORT_ClosePipe(DeviceHandle, FdoDevice, PipeHandle); 1214c2c66affSColin Finck } 1215c2c66affSColin Finck 1216c2c66affSColin Finck KeReleaseSemaphore(&FdoExtension->DeviceSemaphore, 1217c2c66affSColin Finck LOW_REALTIME_PRIORITY, 1218c2c66affSColin Finck 1, 1219c2c66affSColin Finck FALSE); 1220c2c66affSColin Finck 1221c2c66affSColin Finck ExFreePoolWithTag(DeviceHandle, USB_PORT_TAG); 1222c2c66affSColin Finck 1223c2c66affSColin Finck return Status; 1224c2c66affSColin Finck } 1225c2c66affSColin Finck 1226c2c66affSColin Finck ULONG 1227c2c66affSColin Finck NTAPI 1228c2c66affSColin Finck USBPORT_AllocateUsbAddress(IN PDEVICE_OBJECT FdoDevice) 1229c2c66affSColin Finck { 1230c2c66affSColin Finck PUSBPORT_DEVICE_EXTENSION FdoExtension; 1231c2c66affSColin Finck ULONG BitMapIdx; 1232c2c66affSColin Finck ULONG BitNumber; 1233c2c66affSColin Finck ULONG ix; 1234c2c66affSColin Finck 1235c2c66affSColin Finck DPRINT("USBPORT_AllocateUsbAddress \n"); 1236c2c66affSColin Finck 1237c2c66affSColin Finck FdoExtension = FdoDevice->DeviceExtension; 1238c2c66affSColin Finck 1239c2c66affSColin Finck for (ix = 0; ix < 4; ++ix) 1240c2c66affSColin Finck { 1241c2c66affSColin Finck BitMapIdx = 1; 1242c2c66affSColin Finck 1243c2c66affSColin Finck for (BitNumber = 0; BitNumber < 32; ++BitNumber) 1244c2c66affSColin Finck { 1245c2c66affSColin Finck if (!(FdoExtension->UsbAddressBitMap[ix] & BitMapIdx)) 1246c2c66affSColin Finck { 1247c2c66affSColin Finck FdoExtension->UsbAddressBitMap[ix] |= BitMapIdx; 1248c2c66affSColin Finck return 32 * ix + BitNumber; 1249c2c66affSColin Finck } 1250c2c66affSColin Finck 1251c2c66affSColin Finck BitMapIdx <<= 2; 1252c2c66affSColin Finck } 1253c2c66affSColin Finck } 1254c2c66affSColin Finck 1255c2c66affSColin Finck return 0; 1256c2c66affSColin Finck } 1257c2c66affSColin Finck 1258c2c66affSColin Finck VOID 1259c2c66affSColin Finck NTAPI 1260c2c66affSColin Finck USBPORT_FreeUsbAddress(IN PDEVICE_OBJECT FdoDevice, 1261c2c66affSColin Finck IN USHORT DeviceAddress) 1262c2c66affSColin Finck { 1263c2c66affSColin Finck PUSBPORT_DEVICE_EXTENSION FdoExtension; 1264c2c66affSColin Finck ULONG ix; 1265c2c66affSColin Finck ULONG BitMapIdx; 1266c2c66affSColin Finck ULONG BitNumber; 1267c2c66affSColin Finck USHORT CurrentAddress; 1268c2c66affSColin Finck 1269c2c66affSColin Finck DPRINT("USBPORT_FreeUsbAddress: DeviceAddress - %x\n", DeviceAddress); 1270c2c66affSColin Finck 1271c2c66affSColin Finck FdoExtension = FdoDevice->DeviceExtension; 1272c2c66affSColin Finck 1273c2c66affSColin Finck for (ix = 0; ix < 4; ++ix) 1274c2c66affSColin Finck { 1275c2c66affSColin Finck BitMapIdx = 1; 1276c2c66affSColin Finck CurrentAddress = 32 * ix; 1277c2c66affSColin Finck 1278c2c66affSColin Finck for (BitNumber = 0; BitNumber < 32; ++BitNumber) 1279c2c66affSColin Finck { 1280c2c66affSColin Finck if (CurrentAddress == DeviceAddress) 1281c2c66affSColin Finck { 1282c2c66affSColin Finck FdoExtension->UsbAddressBitMap[ix] &= ~BitMapIdx; 1283c2c66affSColin Finck return; 1284c2c66affSColin Finck } 1285c2c66affSColin Finck 1286c2c66affSColin Finck BitMapIdx <<= 2; 1287c2c66affSColin Finck CurrentAddress++; 1288c2c66affSColin Finck } 1289c2c66affSColin Finck } 1290c2c66affSColin Finck } 1291c2c66affSColin Finck 1292c2c66affSColin Finck NTSTATUS 1293c2c66affSColin Finck NTAPI 1294c2c66affSColin Finck USBPORT_InitializeDevice(IN PUSBPORT_DEVICE_HANDLE DeviceHandle, 1295c2c66affSColin Finck IN PDEVICE_OBJECT FdoDevice) 1296c2c66affSColin Finck { 1297c2c66affSColin Finck PUSBPORT_ENDPOINT Endpoint; 1298c2c66affSColin Finck USB_DEFAULT_PIPE_SETUP_PACKET CtrlSetup; 1299c2c66affSColin Finck ULONG TransferedLen; 1300c2c66affSColin Finck USHORT DeviceAddress = 0; 1301c2c66affSColin Finck UCHAR MaxPacketSize; 1302c2c66affSColin Finck NTSTATUS Status; 1303c2c66affSColin Finck PUSBPORT_DEVICE_EXTENSION FdoExtension; 1304c2c66affSColin Finck 1305c2c66affSColin Finck DPRINT("USBPORT_InitializeDevice: ... \n"); 1306c2c66affSColin Finck 1307c2c66affSColin Finck ASSERT(DeviceHandle != NULL); 1308c2c66affSColin Finck 1309c2c66affSColin Finck FdoExtension = FdoDevice->DeviceExtension; 1310c2c66affSColin Finck 1311c2c66affSColin Finck KeWaitForSingleObject(&FdoExtension->DeviceSemaphore, 1312c2c66affSColin Finck Executive, 1313c2c66affSColin Finck KernelMode, 1314c2c66affSColin Finck FALSE, 1315c2c66affSColin Finck NULL); 1316c2c66affSColin Finck 1317c2c66affSColin Finck DeviceAddress = USBPORT_AllocateUsbAddress(FdoDevice); 1318c2c66affSColin Finck ASSERT(DeviceHandle->DeviceAddress == USB_DEFAULT_DEVICE_ADDRESS); 1319c2c66affSColin Finck 1320c2c66affSColin Finck RtlZeroMemory(&CtrlSetup, sizeof(USB_DEFAULT_PIPE_SETUP_PACKET)); 1321c2c66affSColin Finck 1322c2c66affSColin Finck CtrlSetup.bRequest = USB_REQUEST_SET_ADDRESS; 1323c2c66affSColin Finck CtrlSetup.wValue.W = DeviceAddress; 1324c2c66affSColin Finck 1325c2c66affSColin Finck Status = USBPORT_SendSetupPacket(DeviceHandle, 1326c2c66affSColin Finck FdoDevice, 1327c2c66affSColin Finck &CtrlSetup, 1328c2c66affSColin Finck NULL, 1329c2c66affSColin Finck 0, 1330c2c66affSColin Finck NULL, 1331c2c66affSColin Finck NULL); 1332c2c66affSColin Finck 1333c2c66affSColin Finck DPRINT("USBPORT_InitializeDevice: DeviceAddress - %x. SendSetupPacket Status - %x\n", 1334c2c66affSColin Finck DeviceAddress, 1335c2c66affSColin Finck Status); 1336c2c66affSColin Finck 1337c2c66affSColin Finck if (!NT_SUCCESS(Status)) 1338c2c66affSColin Finck goto ExitError; 1339c2c66affSColin Finck 1340c2c66affSColin Finck DeviceHandle->DeviceAddress = DeviceAddress; 1341c2c66affSColin Finck Endpoint = DeviceHandle->PipeHandle.Endpoint; 1342c2c66affSColin Finck 1343c2c66affSColin Finck Endpoint->EndpointProperties.TotalMaxPacketSize = DeviceHandle->DeviceDescriptor.bMaxPacketSize0; 1344c2c66affSColin Finck Endpoint->EndpointProperties.DeviceAddress = DeviceAddress; 1345c2c66affSColin Finck 1346c2c66affSColin Finck Status = USBPORT_ReopenPipe(FdoDevice, Endpoint); 1347c2c66affSColin Finck 1348c2c66affSColin Finck if (!NT_SUCCESS(Status)) 1349c2c66affSColin Finck goto ExitError; 1350c2c66affSColin Finck 1351c2c66affSColin Finck USBPORT_Wait(FdoDevice, 10); 1352c2c66affSColin Finck 1353c2c66affSColin Finck RtlZeroMemory(&CtrlSetup, sizeof(USB_DEFAULT_PIPE_SETUP_PACKET)); 1354c2c66affSColin Finck 1355c2c66affSColin Finck CtrlSetup.bRequest = USB_REQUEST_GET_DESCRIPTOR; 1356c2c66affSColin Finck CtrlSetup.wValue.HiByte = USB_DEVICE_DESCRIPTOR_TYPE; 1357c2c66affSColin Finck CtrlSetup.wLength = sizeof(USB_DEVICE_DESCRIPTOR); 1358c2c66affSColin Finck CtrlSetup.bmRequestType.B = 0x80; 1359c2c66affSColin Finck 1360c2c66affSColin Finck Status = USBPORT_SendSetupPacket(DeviceHandle, 1361c2c66affSColin Finck FdoDevice, 1362c2c66affSColin Finck &CtrlSetup, 1363c2c66affSColin Finck &DeviceHandle->DeviceDescriptor, 1364c2c66affSColin Finck sizeof(USB_DEVICE_DESCRIPTOR), 1365c2c66affSColin Finck &TransferedLen, 1366c2c66affSColin Finck NULL); 1367c2c66affSColin Finck 1368c2c66affSColin Finck if (NT_SUCCESS(Status)) 1369c2c66affSColin Finck { 1370c2c66affSColin Finck ASSERT(TransferedLen == sizeof(USB_DEVICE_DESCRIPTOR)); 1371c2c66affSColin Finck ASSERT(DeviceHandle->DeviceDescriptor.bLength >= sizeof(USB_DEVICE_DESCRIPTOR)); 1372c2c66affSColin Finck ASSERT(DeviceHandle->DeviceDescriptor.bDescriptorType == USB_DEVICE_DESCRIPTOR_TYPE); 1373c2c66affSColin Finck 1374c2c66affSColin Finck MaxPacketSize = DeviceHandle->DeviceDescriptor.bMaxPacketSize0; 1375c2c66affSColin Finck 1376c2c66affSColin Finck ASSERT((MaxPacketSize == 8) || 1377c2c66affSColin Finck (MaxPacketSize == 16) || 1378c2c66affSColin Finck (MaxPacketSize == 32) || 1379c2c66affSColin Finck (MaxPacketSize == 64)); 1380cbba6e80SVadim Galyant 1381cbba6e80SVadim Galyant if (DeviceHandle->DeviceSpeed == UsbHighSpeed && 1382cbba6e80SVadim Galyant DeviceHandle->DeviceDescriptor.bDeviceClass == USB_DEVICE_CLASS_HUB) 1383cbba6e80SVadim Galyant { 1384cbba6e80SVadim Galyant DeviceHandle->Flags |= DEVICE_HANDLE_FLAG_USB2HUB; 1385cbba6e80SVadim Galyant } 1386c2c66affSColin Finck } 1387c2c66affSColin Finck else 1388c2c66affSColin Finck { 1389c2c66affSColin Finck ExitError: 1390c2c66affSColin Finck DPRINT1("USBPORT_InitializeDevice: ExitError. Status - %x\n", Status); 1391c2c66affSColin Finck } 1392c2c66affSColin Finck 1393c2c66affSColin Finck KeReleaseSemaphore(&FdoExtension->DeviceSemaphore, 1394c2c66affSColin Finck LOW_REALTIME_PRIORITY, 1395c2c66affSColin Finck 1, 1396c2c66affSColin Finck FALSE); 1397c2c66affSColin Finck 1398c2c66affSColin Finck return Status; 1399c2c66affSColin Finck } 1400c2c66affSColin Finck 1401c2c66affSColin Finck NTSTATUS 1402c2c66affSColin Finck NTAPI 1403c2c66affSColin Finck USBPORT_GetUsbDescriptor(IN PUSBPORT_DEVICE_HANDLE DeviceHandle, 1404c2c66affSColin Finck IN PDEVICE_OBJECT FdoDevice, 1405c2c66affSColin Finck IN UCHAR Type, 1406c2c66affSColin Finck IN PUCHAR ConfigDesc, 1407c2c66affSColin Finck IN PULONG ConfigDescSize) 1408c2c66affSColin Finck { 1409c2c66affSColin Finck USB_DEFAULT_PIPE_SETUP_PACKET SetupPacket; 1410c2c66affSColin Finck 1411c2c66affSColin Finck DPRINT("USBPORT_GetUsbDescriptor: Type - %x\n"); 1412c2c66affSColin Finck 1413c2c66affSColin Finck RtlZeroMemory(&SetupPacket, sizeof(USB_DEFAULT_PIPE_SETUP_PACKET)); 1414c2c66affSColin Finck 1415c2c66affSColin Finck SetupPacket.bmRequestType.Dir = BMREQUEST_DEVICE_TO_HOST; 1416c2c66affSColin Finck SetupPacket.bRequest = USB_REQUEST_GET_DESCRIPTOR; 1417c2c66affSColin Finck SetupPacket.wValue.HiByte = Type; 1418c2c66affSColin Finck SetupPacket.wLength = (USHORT)*ConfigDescSize; 1419c2c66affSColin Finck 1420c2c66affSColin Finck return USBPORT_SendSetupPacket(DeviceHandle, 1421c2c66affSColin Finck FdoDevice, 1422c2c66affSColin Finck &SetupPacket, 1423c2c66affSColin Finck ConfigDesc, 1424c2c66affSColin Finck *ConfigDescSize, 1425c2c66affSColin Finck ConfigDescSize, 1426c2c66affSColin Finck NULL); 1427c2c66affSColin Finck } 1428c2c66affSColin Finck 1429c2c66affSColin Finck PUSBPORT_INTERFACE_HANDLE 1430c2c66affSColin Finck NTAPI 1431c2c66affSColin Finck USBPORT_GetInterfaceHandle(IN PUSBPORT_CONFIGURATION_HANDLE ConfigurationHandle, 1432c2c66affSColin Finck IN UCHAR InterfaceNumber) 1433c2c66affSColin Finck { 1434c2c66affSColin Finck PUSBPORT_INTERFACE_HANDLE InterfaceHandle; 1435c2c66affSColin Finck PLIST_ENTRY iHandleList; 1436c2c66affSColin Finck UCHAR InterfaceNum; 1437c2c66affSColin Finck 1438c2c66affSColin Finck DPRINT("USBPORT_GetInterfaceHandle: ConfigurationHandle - %p, InterfaceNumber - %p\n", 1439c2c66affSColin Finck ConfigurationHandle, 1440c2c66affSColin Finck InterfaceNumber); 1441c2c66affSColin Finck 1442c2c66affSColin Finck iHandleList = ConfigurationHandle->InterfaceHandleList.Flink; 1443c2c66affSColin Finck 1444c2c66affSColin Finck while (iHandleList && 1445c2c66affSColin Finck (iHandleList != &ConfigurationHandle->InterfaceHandleList)) 1446c2c66affSColin Finck { 1447c2c66affSColin Finck InterfaceHandle = CONTAINING_RECORD(iHandleList, 1448c2c66affSColin Finck USBPORT_INTERFACE_HANDLE, 1449c2c66affSColin Finck InterfaceLink); 1450c2c66affSColin Finck 1451c2c66affSColin Finck InterfaceNum = InterfaceHandle->InterfaceDescriptor.bInterfaceNumber; 1452c2c66affSColin Finck 1453c2c66affSColin Finck if (InterfaceNum == InterfaceNumber) 1454c2c66affSColin Finck return InterfaceHandle; 1455c2c66affSColin Finck 1456c2c66affSColin Finck iHandleList = InterfaceHandle->InterfaceLink.Flink; 1457c2c66affSColin Finck } 1458c2c66affSColin Finck 1459c2c66affSColin Finck return NULL; 1460c2c66affSColin Finck } 1461c2c66affSColin Finck 1462c2c66affSColin Finck NTSTATUS 1463c2c66affSColin Finck NTAPI 1464c2c66affSColin Finck USBPORT_HandleSelectInterface(IN PDEVICE_OBJECT FdoDevice, 1465c2c66affSColin Finck IN PIRP Irp, 1466c2c66affSColin Finck IN PURB Urb) 1467c2c66affSColin Finck { 1468c2c66affSColin Finck PUSBPORT_DEVICE_HANDLE DeviceHandle; 1469c2c66affSColin Finck PUSBPORT_CONFIGURATION_HANDLE ConfigurationHandle; 1470c2c66affSColin Finck PUSBD_INTERFACE_INFORMATION Interface; 1471c2c66affSColin Finck PUSBPORT_INTERFACE_HANDLE InterfaceHandle; 1472c2c66affSColin Finck PUSBPORT_INTERFACE_HANDLE iHandle; 1473c2c66affSColin Finck PUSBPORT_PIPE_HANDLE PipeHandle; 1474c2c66affSColin Finck USBD_STATUS USBDStatus; 1475c2c66affSColin Finck USHORT Length; 1476c2c66affSColin Finck ULONG ix; 1477c2c66affSColin Finck PUSBPORT_DEVICE_EXTENSION FdoExtension; 1478c2c66affSColin Finck 1479c2c66affSColin Finck DPRINT("USBPORT_HandleSelectInterface: ... \n"); 1480c2c66affSColin Finck 1481c2c66affSColin Finck FdoExtension = FdoDevice->DeviceExtension; 1482c2c66affSColin Finck 1483c2c66affSColin Finck KeWaitForSingleObject(&FdoExtension->DeviceSemaphore, 1484c2c66affSColin Finck Executive, 1485c2c66affSColin Finck KernelMode, 1486c2c66affSColin Finck FALSE, 1487c2c66affSColin Finck NULL); 1488c2c66affSColin Finck 1489c2c66affSColin Finck ConfigurationHandle = Urb->UrbSelectInterface.ConfigurationHandle; 1490c2c66affSColin Finck 1491c2c66affSColin Finck Interface = &Urb->UrbSelectInterface.Interface; 1492c2c66affSColin Finck 1493c2c66affSColin Finck Length = Interface->Length + sizeof(USBD_PIPE_INFORMATION); 1494c2c66affSColin Finck Urb->UrbHeader.Length = Length; 1495c2c66affSColin Finck 1496c2c66affSColin Finck USBDStatus = USBPORT_InitInterfaceInfo(Interface, ConfigurationHandle); 1497c2c66affSColin Finck 1498c2c66affSColin Finck if (USBDStatus) 1499c2c66affSColin Finck { 1500c2c66affSColin Finck Interface->InterfaceHandle = (USBD_INTERFACE_HANDLE)-1; 1501c2c66affSColin Finck return USBPORT_USBDStatusToNtStatus(Urb, USBDStatus); 1502c2c66affSColin Finck } 1503c2c66affSColin Finck 1504c2c66affSColin Finck DeviceHandle = Urb->UrbHeader.UsbdDeviceHandle; 1505c2c66affSColin Finck 1506c2c66affSColin Finck InterfaceHandle = USBPORT_GetInterfaceHandle(ConfigurationHandle, 1507c2c66affSColin Finck Interface->InterfaceNumber); 1508c2c66affSColin Finck 1509c2c66affSColin Finck if (InterfaceHandle) 1510c2c66affSColin Finck { 1511c2c66affSColin Finck RemoveEntryList(&InterfaceHandle->InterfaceLink); 1512c2c66affSColin Finck 1513c2c66affSColin Finck if (InterfaceHandle->InterfaceDescriptor.bNumEndpoints) 1514c2c66affSColin Finck { 1515c2c66affSColin Finck PipeHandle = &InterfaceHandle->PipeHandle[0]; 1516c2c66affSColin Finck 1517c2c66affSColin Finck for (ix = 0; 1518c2c66affSColin Finck ix < InterfaceHandle->InterfaceDescriptor.bNumEndpoints; 1519c2c66affSColin Finck ix++) 1520c2c66affSColin Finck { 1521c2c66affSColin Finck USBPORT_ClosePipe(DeviceHandle, FdoDevice, PipeHandle); 1522c2c66affSColin Finck PipeHandle += 1; 1523c2c66affSColin Finck } 1524c2c66affSColin Finck } 1525c2c66affSColin Finck } 1526c2c66affSColin Finck 1527c2c66affSColin Finck iHandle = 0; 1528c2c66affSColin Finck 1529c2c66affSColin Finck USBDStatus = USBPORT_OpenInterface(Urb, 1530c2c66affSColin Finck DeviceHandle, 1531c2c66affSColin Finck FdoDevice, 1532c2c66affSColin Finck ConfigurationHandle, 1533c2c66affSColin Finck Interface, 1534c2c66affSColin Finck &iHandle, 1535c2c66affSColin Finck 1); 1536c2c66affSColin Finck 1537c2c66affSColin Finck if (USBDStatus) 1538c2c66affSColin Finck { 1539c2c66affSColin Finck Interface->InterfaceHandle = (USBD_INTERFACE_HANDLE)-1; 1540c2c66affSColin Finck } 1541c2c66affSColin Finck else 1542c2c66affSColin Finck { 1543c2c66affSColin Finck if (InterfaceHandle) 1544c2c66affSColin Finck ExFreePoolWithTag(InterfaceHandle, USB_PORT_TAG); 1545c2c66affSColin Finck 1546c2c66affSColin Finck Interface->InterfaceHandle = iHandle; 1547c2c66affSColin Finck 1548c2c66affSColin Finck InsertTailList(&ConfigurationHandle->InterfaceHandleList, 1549c2c66affSColin Finck &iHandle->InterfaceLink); 1550c2c66affSColin Finck } 1551c2c66affSColin Finck 1552c2c66affSColin Finck KeReleaseSemaphore(&FdoExtension->DeviceSemaphore, 1553c2c66affSColin Finck LOW_REALTIME_PRIORITY, 1554c2c66affSColin Finck 1, 1555c2c66affSColin Finck FALSE); 1556c2c66affSColin Finck 1557c2c66affSColin Finck return USBPORT_USBDStatusToNtStatus(Urb, USBDStatus); 1558c2c66affSColin Finck } 1559c2c66affSColin Finck 1560c2c66affSColin Finck NTSTATUS 1561c2c66affSColin Finck NTAPI 1562c2c66affSColin Finck USBPORT_RemoveDevice(IN PDEVICE_OBJECT FdoDevice, 1563c2c66affSColin Finck IN OUT PUSBPORT_DEVICE_HANDLE DeviceHandle, 1564c2c66affSColin Finck IN ULONG Flags) 1565c2c66affSColin Finck { 1566c2c66affSColin Finck PUSBPORT_DEVICE_EXTENSION FdoExtension; 15675e15ba48SVadim Galyant PUSB2_TT_EXTENSION TtExtension; 15685e15ba48SVadim Galyant ULONG ix; 15695e15ba48SVadim Galyant KIRQL OldIrql; 1570c2c66affSColin Finck 1571c2c66affSColin Finck DPRINT("USBPORT_RemoveDevice: DeviceHandle - %p, Flags - %x\n", 1572c2c66affSColin Finck DeviceHandle, 1573c2c66affSColin Finck Flags); 1574c2c66affSColin Finck 1575c2c66affSColin Finck FdoExtension = FdoDevice->DeviceExtension; 1576c2c66affSColin Finck 1577c2c66affSColin Finck if ((Flags & USBD_KEEP_DEVICE_DATA) || 1578c2c66affSColin Finck (Flags & USBD_MARK_DEVICE_BUSY)) 1579c2c66affSColin Finck { 1580c2c66affSColin Finck return STATUS_SUCCESS; 1581c2c66affSColin Finck } 1582c2c66affSColin Finck 1583c2c66affSColin Finck KeWaitForSingleObject(&FdoExtension->DeviceSemaphore, 1584c2c66affSColin Finck Executive, 1585c2c66affSColin Finck KernelMode, 1586c2c66affSColin Finck FALSE, 1587c2c66affSColin Finck NULL); 1588c2c66affSColin Finck 1589c2c66affSColin Finck if (!USBPORT_ValidateDeviceHandle(FdoDevice, DeviceHandle)) 1590c2c66affSColin Finck { 1591c2c66affSColin Finck KeReleaseSemaphore(&FdoExtension->DeviceSemaphore, 1592c2c66affSColin Finck LOW_REALTIME_PRIORITY, 1593c2c66affSColin Finck 1, 1594c2c66affSColin Finck FALSE); 1595c2c66affSColin Finck 1596c2c66affSColin Finck DPRINT1("USBPORT_RemoveDevice: Not valid device handle\n"); 1597c2c66affSColin Finck return STATUS_DEVICE_NOT_CONNECTED; 1598c2c66affSColin Finck } 1599c2c66affSColin Finck 1600c2c66affSColin Finck USBPORT_RemoveDeviceHandle(FdoDevice, DeviceHandle); 1601c2c66affSColin Finck 1602c2c66affSColin Finck DeviceHandle->Flags |= DEVICE_HANDLE_FLAG_REMOVED; 1603c2c66affSColin Finck 1604c2c66affSColin Finck USBPORT_AbortTransfers(FdoDevice, DeviceHandle); 1605c2c66affSColin Finck 1606c2c66affSColin Finck DPRINT("USBPORT_RemoveDevice: DeviceHandleLock - %x\n", 1607c2c66affSColin Finck DeviceHandle->DeviceHandleLock); 1608c2c66affSColin Finck 1609c2c66affSColin Finck while (InterlockedDecrement(&DeviceHandle->DeviceHandleLock) >= 0) 1610c2c66affSColin Finck { 1611c2c66affSColin Finck InterlockedIncrement(&DeviceHandle->DeviceHandleLock); 1612c2c66affSColin Finck USBPORT_Wait(FdoDevice, 100); 1613c2c66affSColin Finck } 1614c2c66affSColin Finck 1615c2c66affSColin Finck DPRINT("USBPORT_RemoveDevice: DeviceHandleLock ok\n"); 1616c2c66affSColin Finck 1617c2c66affSColin Finck if (DeviceHandle->ConfigHandle) 1618c2c66affSColin Finck { 1619c2c66affSColin Finck USBPORT_CloseConfiguration(DeviceHandle, FdoDevice); 1620c2c66affSColin Finck } 1621c2c66affSColin Finck 1622c2c66affSColin Finck USBPORT_ClosePipe(DeviceHandle, FdoDevice, &DeviceHandle->PipeHandle); 1623c2c66affSColin Finck 1624c2c66affSColin Finck if (DeviceHandle->DeviceAddress) 1625c2c66affSColin Finck { 1626c2c66affSColin Finck USBPORT_FreeUsbAddress(FdoDevice, DeviceHandle->DeviceAddress); 1627c2c66affSColin Finck } 1628c2c66affSColin Finck 1629*02d1cfd3SVadim Galyant if (!IsListEmpty(&DeviceHandle->TtList)) 1630*02d1cfd3SVadim Galyant { 1631*02d1cfd3SVadim Galyant DPRINT1("USBPORT_RemoveDevice: DeviceHandle->TtList not empty\n"); 1632*02d1cfd3SVadim Galyant } 16335e15ba48SVadim Galyant 16345e15ba48SVadim Galyant while (!IsListEmpty(&DeviceHandle->TtList)) 16355e15ba48SVadim Galyant { 16365e15ba48SVadim Galyant TtExtension = CONTAINING_RECORD(DeviceHandle->TtList.Flink, 16375e15ba48SVadim Galyant USB2_TT_EXTENSION, 16385e15ba48SVadim Galyant Link); 16395e15ba48SVadim Galyant 16405e15ba48SVadim Galyant RemoveHeadList(&DeviceHandle->TtList); 16415e15ba48SVadim Galyant 16425e15ba48SVadim Galyant DPRINT("USBPORT_RemoveDevice: TtExtension - %p\n", TtExtension); 16435e15ba48SVadim Galyant 16445e15ba48SVadim Galyant KeAcquireSpinLock(&FdoExtension->TtSpinLock, &OldIrql); 16455e15ba48SVadim Galyant 16465e15ba48SVadim Galyant TtExtension->Flags |= USB2_TT_EXTENSION_FLAG_DELETED; 16475e15ba48SVadim Galyant 16485e15ba48SVadim Galyant if (IsListEmpty(&TtExtension->EndpointList)) 16495e15ba48SVadim Galyant { 16505e15ba48SVadim Galyant USBPORT_UpdateAllocatedBwTt(TtExtension); 16515e15ba48SVadim Galyant 16525e15ba48SVadim Galyant for (ix = 0; ix < USB2_FRAMES; ix++) 16535e15ba48SVadim Galyant { 16545e15ba48SVadim Galyant FdoExtension->Bandwidth[ix] += TtExtension->MaxBandwidth; 16555e15ba48SVadim Galyant } 16565e15ba48SVadim Galyant 16575e15ba48SVadim Galyant ExFreePool(TtExtension); 16585e15ba48SVadim Galyant } 16595e15ba48SVadim Galyant 16605e15ba48SVadim Galyant KeReleaseSpinLock(&FdoExtension->TtSpinLock, OldIrql); 16615e15ba48SVadim Galyant } 16625e15ba48SVadim Galyant 1663c2c66affSColin Finck KeReleaseSemaphore(&FdoExtension->DeviceSemaphore, 1664c2c66affSColin Finck LOW_REALTIME_PRIORITY, 1665c2c66affSColin Finck 1, 1666c2c66affSColin Finck FALSE); 1667c2c66affSColin Finck 1668c2c66affSColin Finck if (!(DeviceHandle->Flags & DEVICE_HANDLE_FLAG_ROOTHUB)) 1669c2c66affSColin Finck { 1670c2c66affSColin Finck ExFreePoolWithTag(DeviceHandle, USB_PORT_TAG); 1671c2c66affSColin Finck } 1672c2c66affSColin Finck 1673c2c66affSColin Finck return STATUS_SUCCESS; 1674c2c66affSColin Finck } 1675c2c66affSColin Finck 1676c2c66affSColin Finck NTSTATUS 1677c2c66affSColin Finck NTAPI 1678c2c66affSColin Finck USBPORT_RestoreDevice(IN PDEVICE_OBJECT FdoDevice, 1679c2c66affSColin Finck IN OUT PUSBPORT_DEVICE_HANDLE OldDeviceHandle, 1680c2c66affSColin Finck IN OUT PUSBPORT_DEVICE_HANDLE NewDeviceHandle) 1681c2c66affSColin Finck { 1682c2c66affSColin Finck PUSBPORT_DEVICE_EXTENSION FdoExtension; 1683c2c66affSColin Finck PLIST_ENTRY iHandleList; 1684c2c66affSColin Finck PUSBPORT_ENDPOINT Endpoint; 168573bfc3c8SVadim Galyant USBPORT_ENDPOINT_REQUIREMENTS EndpointRequirements = {0}; 1686c2c66affSColin Finck USB_DEFAULT_PIPE_SETUP_PACKET SetupPacket; 1687c2c66affSColin Finck NTSTATUS Status = STATUS_SUCCESS; 1688c2c66affSColin Finck USBD_STATUS USBDStatus; 1689c2c66affSColin Finck KIRQL OldIrql; 1690c2c66affSColin Finck PUSBPORT_INTERFACE_HANDLE InterfaceHandle; 1691c2c66affSColin Finck PUSBPORT_PIPE_HANDLE PipeHandle; 1692c2c66affSColin Finck PUSBPORT_REGISTRATION_PACKET Packet; 1693c2c66affSColin Finck 1694c2c66affSColin Finck DPRINT("USBPORT_RestoreDevice: OldDeviceHandle - %p, NewDeviceHandle - %p\n", 1695c2c66affSColin Finck OldDeviceHandle, 1696c2c66affSColin Finck NewDeviceHandle); 1697c2c66affSColin Finck 1698c2c66affSColin Finck FdoExtension = FdoDevice->DeviceExtension; 1699c2c66affSColin Finck 1700c2c66affSColin Finck KeWaitForSingleObject(&FdoExtension->DeviceSemaphore, 1701c2c66affSColin Finck Executive, 1702c2c66affSColin Finck KernelMode, 1703c2c66affSColin Finck FALSE, 1704c2c66affSColin Finck NULL); 1705c2c66affSColin Finck 1706c2c66affSColin Finck if (!USBPORT_ValidateDeviceHandle(FdoDevice, OldDeviceHandle)) 1707c2c66affSColin Finck { 1708c2c66affSColin Finck KeReleaseSemaphore(&FdoExtension->DeviceSemaphore, 1709c2c66affSColin Finck LOW_REALTIME_PRIORITY, 1710c2c66affSColin Finck 1, 1711c2c66affSColin Finck FALSE); 1712c2c66affSColin Finck 1713c2c66affSColin Finck #ifndef NDEBUG 1714c2c66affSColin Finck DPRINT("USBPORT_RestoreDevice: OldDeviceHandle not valid\n"); 1715c2c66affSColin Finck DbgBreakPoint(); 1716c2c66affSColin Finck #endif 1717c2c66affSColin Finck return STATUS_DEVICE_NOT_CONNECTED; 1718c2c66affSColin Finck } 1719c2c66affSColin Finck 1720c2c66affSColin Finck if (!USBPORT_ValidateDeviceHandle(FdoDevice, NewDeviceHandle)) 1721c2c66affSColin Finck { 1722c2c66affSColin Finck KeReleaseSemaphore(&FdoExtension->DeviceSemaphore, 1723c2c66affSColin Finck LOW_REALTIME_PRIORITY, 1724c2c66affSColin Finck 1, 1725c2c66affSColin Finck FALSE); 1726c2c66affSColin Finck #ifndef NDEBUG 1727c2c66affSColin Finck DPRINT("USBPORT_RestoreDevice: NewDeviceHandle not valid\n"); 1728c2c66affSColin Finck DbgBreakPoint(); 1729c2c66affSColin Finck #endif 1730c2c66affSColin Finck return STATUS_DEVICE_NOT_CONNECTED; 1731c2c66affSColin Finck } 1732c2c66affSColin Finck 1733c2c66affSColin Finck USBPORT_RemoveDeviceHandle(FdoDevice, OldDeviceHandle); 1734c2c66affSColin Finck USBPORT_AbortTransfers(FdoDevice, OldDeviceHandle); 1735c2c66affSColin Finck 1736c2c66affSColin Finck while (InterlockedDecrement(&OldDeviceHandle->DeviceHandleLock) >= 0) 1737c2c66affSColin Finck { 1738c2c66affSColin Finck InterlockedIncrement(&OldDeviceHandle->DeviceHandleLock); 1739c2c66affSColin Finck USBPORT_Wait(FdoDevice, 100); 1740c2c66affSColin Finck } 1741c2c66affSColin Finck 1742c2c66affSColin Finck if (sizeof(USB_DEVICE_DESCRIPTOR) == RtlCompareMemory(&NewDeviceHandle->DeviceDescriptor, 1743c2c66affSColin Finck &OldDeviceHandle->DeviceDescriptor, 1744c2c66affSColin Finck sizeof(USB_DEVICE_DESCRIPTOR))) 1745c2c66affSColin Finck { 1746c2c66affSColin Finck NewDeviceHandle->ConfigHandle = OldDeviceHandle->ConfigHandle; 1747c2c66affSColin Finck 1748c2c66affSColin Finck if (OldDeviceHandle->ConfigHandle) 1749c2c66affSColin Finck { 1750c2c66affSColin Finck RtlZeroMemory(&SetupPacket, sizeof(USB_DEFAULT_PIPE_SETUP_PACKET)); 1751c2c66affSColin Finck 1752c2c66affSColin Finck SetupPacket.bmRequestType.Dir = BMREQUEST_HOST_TO_DEVICE; 1753c2c66affSColin Finck SetupPacket.bRequest = USB_REQUEST_SET_CONFIGURATION; 1754c2c66affSColin Finck SetupPacket.wValue.W = OldDeviceHandle->ConfigHandle->ConfigurationDescriptor->bConfigurationValue; 1755c2c66affSColin Finck SetupPacket.wIndex.W = 0; 1756c2c66affSColin Finck SetupPacket.wLength = 0; 1757c2c66affSColin Finck 1758c2c66affSColin Finck USBPORT_SendSetupPacket(NewDeviceHandle, 1759c2c66affSColin Finck FdoDevice, 1760c2c66affSColin Finck &SetupPacket, 1761c2c66affSColin Finck NULL, 1762c2c66affSColin Finck 0, 1763c2c66affSColin Finck NULL, 1764c2c66affSColin Finck &USBDStatus); 1765c2c66affSColin Finck 1766c2c66affSColin Finck if (USBD_ERROR(USBDStatus)) 1767c2c66affSColin Finck Status = USBPORT_USBDStatusToNtStatus(NULL, USBDStatus); 1768c2c66affSColin Finck 1769c2c66affSColin Finck if (NT_SUCCESS(Status)) 1770c2c66affSColin Finck { 1771c2c66affSColin Finck iHandleList = NewDeviceHandle->ConfigHandle->InterfaceHandleList.Flink; 1772c2c66affSColin Finck 1773c2c66affSColin Finck while (iHandleList && 1774c2c66affSColin Finck iHandleList != &NewDeviceHandle->ConfigHandle->InterfaceHandleList) 1775c2c66affSColin Finck { 1776c2c66affSColin Finck InterfaceHandle = CONTAINING_RECORD(iHandleList, 1777c2c66affSColin Finck USBPORT_INTERFACE_HANDLE, 1778c2c66affSColin Finck InterfaceLink); 1779c2c66affSColin Finck 1780c2c66affSColin Finck if (InterfaceHandle->AlternateSetting) 1781c2c66affSColin Finck { 1782c2c66affSColin Finck RtlZeroMemory(&SetupPacket, sizeof(USB_DEFAULT_PIPE_SETUP_PACKET)); 1783c2c66affSColin Finck 1784c2c66affSColin Finck SetupPacket.bmRequestType.Dir = BMREQUEST_HOST_TO_DEVICE; 1785c2c66affSColin Finck SetupPacket.bmRequestType.Type = BMREQUEST_STANDARD; 1786c2c66affSColin Finck SetupPacket.bmRequestType.Recipient = BMREQUEST_TO_INTERFACE; 1787c2c66affSColin Finck 1788c2c66affSColin Finck SetupPacket.bRequest = USB_REQUEST_SET_INTERFACE; 1789c2c66affSColin Finck SetupPacket.wValue.W = InterfaceHandle->InterfaceDescriptor.bAlternateSetting; 1790c2c66affSColin Finck SetupPacket.wIndex.W = InterfaceHandle->InterfaceDescriptor.bInterfaceNumber; 1791c2c66affSColin Finck SetupPacket.wLength = 0; 1792c2c66affSColin Finck 1793c2c66affSColin Finck USBPORT_SendSetupPacket(NewDeviceHandle, 1794c2c66affSColin Finck FdoDevice, 1795c2c66affSColin Finck &SetupPacket, 1796c2c66affSColin Finck NULL, 1797c2c66affSColin Finck 0, 1798c2c66affSColin Finck NULL, 1799c2c66affSColin Finck &USBDStatus); 1800c2c66affSColin Finck } 1801c2c66affSColin Finck 1802c2c66affSColin Finck iHandleList = iHandleList->Flink; 1803c2c66affSColin Finck } 1804c2c66affSColin Finck } 1805c2c66affSColin Finck } 1806c2c66affSColin Finck 1807cbba6e80SVadim Galyant if (NewDeviceHandle->Flags & DEVICE_HANDLE_FLAG_USB2HUB) 1808c2c66affSColin Finck { 1809c2c66affSColin Finck DPRINT1("USBPORT_RestoreDevice: FIXME Transaction Translator\n"); 1810c2c66affSColin Finck NewDeviceHandle->TtCount = OldDeviceHandle->TtCount; 1811cbba6e80SVadim Galyant 1812cbba6e80SVadim Galyant #ifndef NDEBUG 1813cbba6e80SVadim Galyant DbgBreakPoint(); 1814cbba6e80SVadim Galyant #endif 1815c2c66affSColin Finck } 1816c2c66affSColin Finck 1817c2c66affSColin Finck while (!IsListEmpty(&OldDeviceHandle->PipeHandleList)) 1818c2c66affSColin Finck { 1819c2c66affSColin Finck PipeHandle = CONTAINING_RECORD(OldDeviceHandle->PipeHandleList.Flink, 1820c2c66affSColin Finck USBPORT_PIPE_HANDLE, 1821c2c66affSColin Finck PipeLink); 1822c2c66affSColin Finck 1823c2c66affSColin Finck DPRINT("USBPORT_RestoreDevice: PipeHandle - %p\n", PipeHandle); 1824c2c66affSColin Finck 1825c2c66affSColin Finck USBPORT_RemovePipeHandle(OldDeviceHandle, PipeHandle); 1826c2c66affSColin Finck 1827c2c66affSColin Finck if (PipeHandle != &OldDeviceHandle->PipeHandle) 1828c2c66affSColin Finck { 1829c2c66affSColin Finck USBPORT_AddPipeHandle(NewDeviceHandle, PipeHandle); 1830c2c66affSColin Finck 1831c2c66affSColin Finck if (!(PipeHandle->Flags & PIPE_HANDLE_FLAG_NULL_PACKET_SIZE)) 1832c2c66affSColin Finck { 1833c2c66affSColin Finck Endpoint = PipeHandle->Endpoint; 1834c2c66affSColin Finck Endpoint->DeviceHandle = NewDeviceHandle; 1835c2c66affSColin Finck Endpoint->EndpointProperties.DeviceAddress = NewDeviceHandle->DeviceAddress; 1836c2c66affSColin Finck 1837c2c66affSColin Finck Packet = &FdoExtension->MiniPortInterface->Packet; 1838c2c66affSColin Finck 1839c2c66affSColin Finck if (!(Endpoint->Flags & ENDPOINT_FLAG_NUKE)) 1840c2c66affSColin Finck { 1841c2c66affSColin Finck KeAcquireSpinLock(&FdoExtension->MiniportSpinLock, 1842c2c66affSColin Finck &OldIrql); 1843c2c66affSColin Finck 1844c2c66affSColin Finck Packet->ReopenEndpoint(FdoExtension->MiniPortExt, 1845c2c66affSColin Finck &Endpoint->EndpointProperties, 1846c2c66affSColin Finck Endpoint + 1); 1847c2c66affSColin Finck 1848c2c66affSColin Finck Packet->SetEndpointDataToggle(FdoExtension->MiniPortExt, 1849c2c66affSColin Finck Endpoint + 1, 1850c2c66affSColin Finck 0); 1851c2c66affSColin Finck 1852c2c66affSColin Finck Packet->SetEndpointStatus(FdoExtension->MiniPortExt, 1853c2c66affSColin Finck Endpoint + 1, 1854c2c66affSColin Finck USBPORT_ENDPOINT_RUN); 1855c2c66affSColin Finck 1856c2c66affSColin Finck KeReleaseSpinLock(&FdoExtension->MiniportSpinLock, 1857c2c66affSColin Finck OldIrql); 1858c2c66affSColin Finck } 1859c2c66affSColin Finck else 1860c2c66affSColin Finck { 1861c2c66affSColin Finck MiniportCloseEndpoint(FdoDevice, Endpoint); 1862c2c66affSColin Finck 1863c2c66affSColin Finck RtlZeroMemory(Endpoint + 1, Packet->MiniPortEndpointSize); 1864c2c66affSColin Finck 1865c2c66affSColin Finck RtlZeroMemory((PVOID)Endpoint->EndpointProperties.BufferVA, 1866c2c66affSColin Finck Endpoint->EndpointProperties.BufferLength); 1867c2c66affSColin Finck 1868c2c66affSColin Finck KeAcquireSpinLock(&FdoExtension->MiniportSpinLock, &OldIrql); 1869c2c66affSColin Finck 1870c2c66affSColin Finck Packet->QueryEndpointRequirements(FdoExtension->MiniPortExt, 1871c2c66affSColin Finck &Endpoint->EndpointProperties, 187273bfc3c8SVadim Galyant &EndpointRequirements); 1873c2c66affSColin Finck 1874c2c66affSColin Finck KeReleaseSpinLock(&FdoExtension->MiniportSpinLock, 1875c2c66affSColin Finck OldIrql); 1876c2c66affSColin Finck 1877c2c66affSColin Finck MiniportOpenEndpoint(FdoDevice, Endpoint); 1878c2c66affSColin Finck 1879c2c66affSColin Finck Endpoint->Flags &= ~(ENDPOINT_FLAG_NUKE | 1880c2c66affSColin Finck ENDPOINT_FLAG_ABORTING); 1881c2c66affSColin Finck 1882c2c66affSColin Finck KeAcquireSpinLock(&Endpoint->EndpointSpinLock, 1883c2c66affSColin Finck &Endpoint->EndpointOldIrql); 1884c2c66affSColin Finck 1885c2c66affSColin Finck if (Endpoint->StateLast == USBPORT_ENDPOINT_ACTIVE) 1886c2c66affSColin Finck { 1887c2c66affSColin Finck KeAcquireSpinLockAtDpcLevel(&FdoExtension->MiniportSpinLock); 1888c2c66affSColin Finck 1889c2c66affSColin Finck Packet->SetEndpointState(FdoExtension->MiniPortExt, 1890c2c66affSColin Finck Endpoint + 1, 1891c2c66affSColin Finck USBPORT_ENDPOINT_ACTIVE); 1892c2c66affSColin Finck 1893c2c66affSColin Finck KeReleaseSpinLockFromDpcLevel(&FdoExtension->MiniportSpinLock); 1894c2c66affSColin Finck } 1895c2c66affSColin Finck 1896c2c66affSColin Finck KeReleaseSpinLock(&Endpoint->EndpointSpinLock, 1897c2c66affSColin Finck Endpoint->EndpointOldIrql); 1898c2c66affSColin Finck } 1899c2c66affSColin Finck } 1900c2c66affSColin Finck } 1901c2c66affSColin Finck } 1902c2c66affSColin Finck 1903c2c66affSColin Finck USBPORT_AddPipeHandle(OldDeviceHandle, &OldDeviceHandle->PipeHandle); 1904c2c66affSColin Finck } 1905c2c66affSColin Finck else 1906c2c66affSColin Finck { 1907c2c66affSColin Finck #ifndef NDEBUG 1908c2c66affSColin Finck DPRINT("USBPORT_RestoreDevice: New DeviceDescriptor != Old DeviceDescriptor\n"); 1909c2c66affSColin Finck DbgBreakPoint(); 1910c2c66affSColin Finck #endif 1911c2c66affSColin Finck Status = STATUS_UNSUCCESSFUL; 1912c2c66affSColin Finck } 1913c2c66affSColin Finck 1914c2c66affSColin Finck USBPORT_ClosePipe(OldDeviceHandle, FdoDevice, &OldDeviceHandle->PipeHandle); 1915c2c66affSColin Finck 1916c2c66affSColin Finck if (OldDeviceHandle->DeviceAddress != 0) 1917c2c66affSColin Finck USBPORT_FreeUsbAddress(FdoDevice, OldDeviceHandle->DeviceAddress); 1918c2c66affSColin Finck 1919c2c66affSColin Finck KeReleaseSemaphore(&FdoExtension->DeviceSemaphore, 1920c2c66affSColin Finck LOW_REALTIME_PRIORITY, 1921c2c66affSColin Finck 1, 1922c2c66affSColin Finck FALSE); 1923c2c66affSColin Finck 1924c2c66affSColin Finck ExFreePoolWithTag(OldDeviceHandle, USB_PORT_TAG); 1925c2c66affSColin Finck 1926c2c66affSColin Finck return Status; 1927c2c66affSColin Finck } 1928c2c66affSColin Finck 1929c2c66affSColin Finck NTSTATUS 1930c2c66affSColin Finck NTAPI 1931c2c66affSColin Finck USBPORT_InitializeTT(IN PDEVICE_OBJECT FdoDevice, 1932c2c66affSColin Finck IN PUSBPORT_DEVICE_HANDLE HubDeviceHandle, 1933c2c66affSColin Finck IN ULONG TtNumber) 1934c2c66affSColin Finck { 193518702047SVadim Galyant PUSBPORT_DEVICE_EXTENSION FdoExtension; 193618702047SVadim Galyant PUSB2_TT_EXTENSION TtExtension; 193718702047SVadim Galyant ULONG ix; 193818702047SVadim Galyant 193918702047SVadim Galyant DPRINT("USBPORT_InitializeTT: HubDeviceHandle - %p, TtNumber - %X\n", 194018702047SVadim Galyant HubDeviceHandle, 194118702047SVadim Galyant TtNumber); 194218702047SVadim Galyant 194318702047SVadim Galyant FdoExtension = FdoDevice->DeviceExtension; 194418702047SVadim Galyant 194518702047SVadim Galyant TtExtension = ExAllocatePoolWithTag(NonPagedPool, 194618702047SVadim Galyant sizeof(USB2_TT_EXTENSION), 194718702047SVadim Galyant USB_PORT_TAG); 194818702047SVadim Galyant 194918702047SVadim Galyant if (!TtExtension) 195018702047SVadim Galyant { 195118702047SVadim Galyant DPRINT1("USBPORT_InitializeTT: ExAllocatePoolWithTag return NULL\n"); 195218702047SVadim Galyant return STATUS_INSUFFICIENT_RESOURCES; 195318702047SVadim Galyant } 195418702047SVadim Galyant 195518702047SVadim Galyant DPRINT("USBPORT_InitializeTT: TtExtension - %p\n", TtExtension); 195618702047SVadim Galyant 195718702047SVadim Galyant RtlZeroMemory(TtExtension, sizeof(USB2_TT_EXTENSION)); 195818702047SVadim Galyant 195918702047SVadim Galyant TtExtension->DeviceAddress = HubDeviceHandle->DeviceAddress; 196018702047SVadim Galyant TtExtension->TtNumber = TtNumber; 196118702047SVadim Galyant TtExtension->RootHubPdo = FdoExtension->RootHubPdo; 196218702047SVadim Galyant TtExtension->BusBandwidth = TOTAL_USB11_BUS_BANDWIDTH; 196318702047SVadim Galyant 1964c6ca2a8fSVadim Galyant InitializeListHead(&TtExtension->EndpointList); 196518702047SVadim Galyant 196618702047SVadim Galyant /* 90% maximum allowed for periodic endpoints */ 196718702047SVadim Galyant for (ix = 0; ix < USB2_FRAMES; ix++) 196818702047SVadim Galyant { 196918702047SVadim Galyant TtExtension->Bandwidth[ix] = TtExtension->BusBandwidth - 197018702047SVadim Galyant TtExtension->BusBandwidth / 10; 197118702047SVadim Galyant } 197218702047SVadim Galyant 197318702047SVadim Galyant USBPORT_UpdateAllocatedBwTt(TtExtension); 197418702047SVadim Galyant 197518702047SVadim Galyant for (ix = 0; ix < USB2_FRAMES; ix++) 197618702047SVadim Galyant { 197718702047SVadim Galyant FdoExtension->Bandwidth[ix] -= TtExtension->MaxBandwidth; 197818702047SVadim Galyant } 197918702047SVadim Galyant 198018702047SVadim Galyant USB2_InitTT(FdoExtension->Usb2Extension, &TtExtension->Tt); 198118702047SVadim Galyant 198218702047SVadim Galyant InsertTailList(&HubDeviceHandle->TtList, &TtExtension->Link); 198318702047SVadim Galyant 1984c2c66affSColin Finck return STATUS_SUCCESS; 1985c2c66affSColin Finck } 1986c2c66affSColin Finck 1987c2c66affSColin Finck NTSTATUS 1988c2c66affSColin Finck NTAPI 1989c2c66affSColin Finck USBPORT_Initialize20Hub(IN PDEVICE_OBJECT FdoDevice, 1990c2c66affSColin Finck IN PUSBPORT_DEVICE_HANDLE HubDeviceHandle, 1991c2c66affSColin Finck IN ULONG TtCount) 1992c2c66affSColin Finck { 1993c2c66affSColin Finck NTSTATUS Status; 1994c2c66affSColin Finck ULONG ix; 1995c2c66affSColin Finck 199618702047SVadim Galyant DPRINT("USBPORT_Initialize20Hub: TtCount - %X\n", TtCount); 1997c2c66affSColin Finck 1998c2c66affSColin Finck if (!HubDeviceHandle) 1999c2c66affSColin Finck { 2000c2c66affSColin Finck return STATUS_INVALID_PARAMETER; 2001c2c66affSColin Finck } 2002c2c66affSColin Finck 2003c2c66affSColin Finck if (HubDeviceHandle->Flags & DEVICE_HANDLE_FLAG_ROOTHUB) 2004c2c66affSColin Finck { 2005c2c66affSColin Finck return STATUS_SUCCESS; 2006c2c66affSColin Finck } 2007c2c66affSColin Finck 2008c2c66affSColin Finck if (TtCount == 0) 2009c2c66affSColin Finck { 2010c2c66affSColin Finck HubDeviceHandle->TtCount = 0; 2011c2c66affSColin Finck return STATUS_SUCCESS; 2012c2c66affSColin Finck } 2013c2c66affSColin Finck 2014c2c66affSColin Finck for (ix = 0; ix < TtCount; ++ix) 2015c2c66affSColin Finck { 2016c2c66affSColin Finck Status = USBPORT_InitializeTT(FdoDevice, HubDeviceHandle, ix + 1); 2017c2c66affSColin Finck 2018c2c66affSColin Finck if (!NT_SUCCESS(Status)) 2019c2c66affSColin Finck break; 2020c2c66affSColin Finck } 2021c2c66affSColin Finck 2022c2c66affSColin Finck HubDeviceHandle->TtCount = TtCount; 2023c2c66affSColin Finck 2024c2c66affSColin Finck return Status; 2025c2c66affSColin Finck }