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

2477 lines
69 KiB

/*++
Copyright (c) 1995,1996 Microsoft Corporation
:ts=4
Module Name:
urb.c
Abstract:
The module manages transactions on the USB.
Environment:
kernel mode only
Notes:
Revision History:
11-01-95 : created
04-28-96 : linked urb support & cancel support
--*/
#include "wdm.h"
#include "stdarg.h"
#include "stdio.h"
#include "usbdi.h"
#include "hcdi.h"
#include "uhcd.h"
#define COMPUTE_HCD_EXTENSION_SIZE(urb) (sizeof(HCD_EXTENSION) + \
(urb->HcdUrbCommonTransfer.TransferBufferLength / PAGE_SIZE + 1) \
* sizeof(UHCD_LOGICAL_ADDRESS))
#define UHCD_IS_TRANSFER(urb) (((urb)->UrbHeader.Function == URB_FUNCTION_CONTROL_TRANSFER || \
(urb)->UrbHeader.Function == URB_FUNCTION_BULK_OR_INTERRUPT_TRANSFER ||\
(urb)->UrbHeader.Function == URB_FUNCTION_ISOCH_TRANSFER) \
? TRUE : FALSE)
// calc the maxrequests based on endpoint type nad
// flag options
UCHAR
MAX_REQUESTS(
IN PUSB_ENDPOINT_DESCRIPTOR EndpointDescriptor,
IN ULONG EpFlags
)
{
if (EpFlags & EPFLAG_DBL_BUFFER) {
return 1;
}
switch (USB_ENDPOINT_TYPE_MASK &
EndpointDescriptor->bmAttributes) {
case USB_ENDPOINT_TYPE_BULK:
return 2;
//return 1;
case USB_ENDPOINT_TYPE_ISOCHRONOUS:
return 3;
default:
return 1;
}
}
#if 0
ULONG
UHCD_CountPageCrossings(
IN ULONG MaxRequests,
IN ULONG MaxTransferSize
)
/*++
Routine Description:
Completes an I/O Request
Arguments:
Return Value:
maximum number of possible page crossings
--*/
{
ULONG pageCrossings;
pageCrossings = (MaxTransferSize + PAGE_SIZE) / PAGE_SIZE;
if (MaxRequests>1) {
pageCrossings *= 2;
}
// now allocate space for packet buffers
UHCD_KdPrint((2, "'UHCD_CountPageCrossings, max transfer size 0x%x -- page crossings = 0x%x\n",
MaxTransferSize, pageCrossings));
UHCD_ASSERT(pageCrossings > 0);
return pageCrossings;
}
#endif
VOID
UHCD_CompleteIrp(
IN PDEVICE_OBJECT DeviceObject,
IN PIRP Irp,
IN NTSTATUS ntStatus,
IN ULONG Information,
IN PHCD_URB Urb
)
/*++
Routine Description:
Completes an I/O Request
Arguments:
DeviceObject - pointer to a device object
Irp - pointer to an I/O Request Packet to complete
ntStatus - status code to set int the IRP when completed
Information -
Urb - root transfer urb if this Irp is associated with
a transfer.
Return Value:
--*/
{
KIRQL irql;
UCHAR flags = 0;
STARTPROC("cIrp");
UHCD_KdPrint((2, "'UHCD_CompleteIrp status = %x\n", ntStatus));
LOGENTRY(LOG_MISC, 'ICan', Urb, 0, Irp);
if (Urb) {
//
// if we have any working space free it now
//
if (UHCD_IS_TRANSFER(Urb) && HCD_AREA(Urb).HcdExtension) {
PHCD_EXTENSION urbWork;
urbWork = HCD_AREA(Urb).HcdExtension;
// remember the flags
flags = urbWork->Flags;
RETHEAP(HCD_AREA(Urb).HcdExtension);
HCD_AREA(Urb).HcdExtension = NULL;
}
//
// Decrement the urb counter that we keep in our stack location for
// the irp, when it goes to zero -- complete it
//
//
// one Urb completed
//
DECREMENT_PENDING_URB_COUNT(Irp);
if (PENDING_URB_COUNT(Irp)) {
//
// stall completion until all Urbs are done.
//
TEST_TRAP();
return;
} else {
IoAcquireCancelSpinLock(&irql);
if (Irp->Cancel) {
LOGENTRY(LOG_MISC, 'irpX', flags, 0, Irp);
// note that the cancel routine will only mess
// with the urbs on the active list -- any active
// urbs should have been removed from the active
// list before calling this routine
//
// Irp associated with this transfer has
// been canceled.
//
// The cancel routine will complete the Irp
// unless there are active transfers.
//
if (flags & UHCD_TRANSFER_ACTIVE) {
IoSetCancelRoutine(Irp, NULL);
IoReleaseCancelSpinLock(irql);
//
// if the io has already started the we must
// complete the Irp with STATUS_CANCELLED here.
//
ntStatus = STATUS_CANCELLED;
Information = 0;
goto UHCD_CompleteIrp_CompleteRequest;
}
IoReleaseCancelSpinLock(irql);
return;
} else {
//
// Irp is no longer cancelable
//
LOGENTRY(LOG_MISC, 'NCan', flags, 0, Irp);
IoSetCancelRoutine(Irp, NULL);
IoReleaseCancelSpinLock(irql);
//
// Pending bit should be cleared
//
UHCD_ASSERT(!USBD_PENDING(Urb->HcdUrbCommonTransfer.Status));
}
}
}
UHCD_CompleteIrp_CompleteRequest:
Irp->IoStatus.Status = ntStatus;
Irp->IoStatus.Information = Information;
LOGENTRY(LOG_MISC, 'irpC', Irp, DeviceObject, Urb);
USBD_CompleteRequest(Irp,
IO_NO_INCREMENT);
ENDPROC("cIrp");
}
NTSTATUS
UHCD_URB_Dispatch(
IN PDEVICE_OBJECT DeviceObject,
IN PIRP Irp
)
/*++
Routine Description:
Process URBs from the dispatch routine.
Arguments:
DeviceObject - pointer to a device object
Irp - pointer to an I/O Request Packet
Return Value:
--*/
{
PHCD_URB urb;
NTSTATUS ntStatus = STATUS_SUCCESS;
PDEVICE_EXTENSION deviceExtension;
PUHCD_ENDPOINT endpoint = NULL;
ULONG siz, cnt;
ULONG numTDs;
UHCD_KdPrint((2, "'enter UHCD_URB_Dispatch \n"));
deviceExtension = (PDEVICE_EXTENSION) DeviceObject->DeviceExtension;
urb = (PHCD_URB) URB_FROM_IRP(Irp);
if (deviceExtension->CurrentDevicePowerState != PowerDeviceD0 ||
deviceExtension->HcFlags & HCFLAG_HCD_SHUTDOWN) {
//
// someone is submitting requests while the
// HC is suspended or OFF, we will just fail them
//
UHCD_KdPrint
((0, "'Warning: UHCD got a request while not in D0 in shutdown\n"));
ntStatus = STATUS_DEVICE_NOT_READY;
URB_HEADER(urb).Status = USBD_STATUS_REQUEST_FAILED;
UHCD_CompleteIrp(DeviceObject, Irp, ntStatus, 0, NULL);
return ntStatus;
}
switch(URB_HEADER(urb).Function) {
//
// These commands get queued to the startio routine
// for processing.
//
case URB_FUNCTION_HCD_OPEN_ENDPOINT:
#ifdef ROOT_HUB
if (urb->HcdUrbOpenEndpoint.DeviceAddress ==
deviceExtension->RootHubDeviceAddress) {
// This routine will either complete
// the irp or mark it pending.
ntStatus = UHCD_RootHub_OpenEndpoint(DeviceObject,
Irp,
urb);
break;
}
#endif
//
// Grow the common buffer pool based on how much
// memory we'll need for transfers
//
// Well need enough for one set of TDs plus
// a common buffer for each possible page crossing
// within a trasnsfer.
// first allocate space for the TDs
// for now use a global value
//
// allocate the endpoint structures here while we are
// at passive level
//
endpoint = (PUHCD_ENDPOINT) GETHEAP(NonPagedPool,
sizeof(UHCD_ENDPOINT));
if (endpoint) {
// start endpoint initialization
RtlZeroMemory(endpoint, sizeof(*endpoint));
endpoint->Sig = SIG_EP;
// check for bulk and iso special options
// we only support double buffering for bulk-IN
// we only support fast iso for iso-OUT
if (urb->HcdUrbOpenEndpoint.HcdEndpointFlags &
USBD_EP_FLAG_DOUBLE_BUFFER) {
if (USB_ENDPOINT_DIRECTION_IN(
urb->HcdUrbOpenEndpoint.EndpointDescriptor->bEndpointAddress)
&&
(USB_ENDPOINT_TYPE_MASK &
urb->HcdUrbOpenEndpoint.EndpointDescriptor->bmAttributes)
==
USB_ENDPOINT_TYPE_BULK) {
SET_EPFLAG(endpoint, EPFLAG_DBL_BUFFER);
} else { // bugbug error here?
UHCD_KdPrint((1, "'WARNING: Cannot double buffer this endpoint\n"));
}
}
#ifdef FAST_ISO
if (urb->HcdUrbOpenEndpoint.HcdEndpointFlags &
USBD_EP_FLAG_FAST_ISO) {
if (USB_ENDPOINT_DIRECTION_OUT(
urb->HcdUrbOpenEndpoint.EndpointDescriptor->bEndpointAddress)
&&
(USB_ENDPOINT_TYPE_MASK &
urb->HcdUrbOpenEndpoint.EndpointDescriptor->bmAttributes)
==
USB_ENDPOINT_TYPE_ISOCHRONOUS) {
SET_EPFLAG(endpoint, EPFLAG_FAST_ISO);
} else { // bugbug error here?
UHCD_KdPrint((1, "'WARNING: Cannot fast-iso this endpoint\n"));
}
}
#endif
urb->HcdUrbOpenEndpoint.HcdEndpoint = endpoint;
urb->HcdUrbOpenEndpoint.ScheduleOffset = 0;
numTDs =
endpoint->TDCount = UHCD_GetNumTDsPerEndoint((UCHAR) (USB_ENDPOINT_TYPE_MASK &
urb->HcdUrbOpenEndpoint.EndpointDescriptor->bmAttributes));
UHCD_ASSERT(TD_LIST_SIZE(numTDs) <= UHCD_LARGE_COMMON_BUFFER_SIZE);
if (endpoint->EndpointFlags & EPFLAG_DBL_BUFFER) {
// do the double buffer init
ntStatus = UHCD_InitializeNoDMAEndpoint(DeviceObject,
endpoint);
}
if (NT_SUCCESS(ntStatus)) {
endpoint->MaxRequests =
MAX_REQUESTS(urb->HcdUrbOpenEndpoint.EndpointDescriptor,
endpoint->EndpointFlags);
UHCD_KdPrint((2, "'MaxRequests = %d\n", endpoint->MaxRequests));
// init transfer sequence numbers
endpoint->NextXferId = 0;
endpoint->CurrentXferId = 0;
if (endpoint->EndpointFlags & EPFLAG_FAST_ISO) {
// do the double buffer init
UHCD_KdPrint((1, "'Using Fast ISO for Endpoint\n"));
ntStatus = UHCD_FinishInitializeEndpoint(DeviceObject,
endpoint,
urb->HcdUrbOpenEndpoint.EndpointDescriptor,
urb);
if (NT_SUCCESS(ntStatus)) {
ntStatus = UHCD_InitializeFastIsoEndpoint(DeviceObject,
endpoint);
}
} else {
for (cnt=0; cnt< endpoint->MaxRequests; cnt++) {
if ((UHCD_AllocateHardwareDescriptors(DeviceObject,
&endpoint->HardwareDescriptorList[cnt],
endpoint->TDCount))) {
PUCHAR descriptors;
ULONG i;
descriptors = endpoint->HardwareDescriptorList[cnt]->MemoryDescriptor->VirtualAddress;
// Initialize the queue head for this endpoint
// use the TD on the first list
if (cnt == 0) {
endpoint->QueueHead = (PHW_QUEUE_HEAD) descriptors;
// save our endpoint in the QueueHead
endpoint->QueueHead->Endpoint = endpoint;
}
//
// the TDs we'll need to service this endpoint
//
endpoint->SlotTDList[cnt] = (PUHCD_TD_LIST) (descriptors + sizeof(HW_QUEUE_HEAD));
// BUGBUG possibly move this to allocate TD code
for (i=0; i<= endpoint->TDCount; i++) {
// one time init stuff
// for isoch TDs.
endpoint->SlotTDList[cnt]->TDs[i].Frame = 0;
endpoint->SlotTDList[cnt]->TDs[i].Urb = NULL;
}
} else {
RETHEAP(endpoint);
endpoint = NULL;
ntStatus = STATUS_INSUFFICIENT_RESOURCES;
break;
}
} // for
// point to first list
if (endpoint) {
endpoint->TDList = endpoint->SlotTDList[0];
}
}
}
} else {
// failed to alloc endpoint structure
UHCD_KdTrap(("UHCD failed to alloc endpoint structure\n"));
ntStatus = STATUS_INSUFFICIENT_RESOURCES;
}
#if DBG
UHCD_KdPrint((2, "'Open Endpoint, max transfer size 0x%x \n",
urb->HcdUrbOpenEndpoint.MaxTransferSize));
// transfers >64k are valid
// if (urb->HcdUrbOpenEndpoint.MaxTransferSize >
// 1024*64) {
// // bigger than 64k, probably a bug.
// UHCD_KdPrint((2, "'Open Endpoint, max transfer size 0x%x \n",
// urb->HcdUrbOpenEndpoint.MaxTransferSize));
// TEST_TRAP();
// }
#endif
if (NT_SUCCESS(ntStatus)) {
//
// take the opportunity to grow our pool
// in necessary
//
// BUGBUG possibly use max_packet size as a hint
UHCD_MoreCommonBuffers(DeviceObject);
URB_HEADER(urb).Status = UHCD_STATUS_PENDING_STARTIO;
ntStatus = STATUS_PENDING;
UHCD_KdPrint((2, "'Queue Irp To StartIo\n"));
Irp->IoStatus.Status = ntStatus;
Irp->IoStatus.Information = 0;
IoMarkIrpPending(Irp);
IoStartPacket(DeviceObject,
Irp,
0,
UHCD_StartIoCancel);
} else {
URB_HEADER(urb).Status = USBD_STATUS_NO_MEMORY;
UHCD_CompleteIrp(DeviceObject, Irp, ntStatus, 0, NULL);
}
break;
case URB_FUNCTION_CONTROL_TRANSFER:
case URB_FUNCTION_BULK_OR_INTERRUPT_TRANSFER:
// this will activate the rolover interrupts
deviceExtension->XferIdleTime = 0;
case URB_FUNCTION_HCD_CLOSE_ENDPOINT:
if (URB_HEADER(urb).Function == URB_FUNCTION_HCD_CLOSE_ENDPOINT) {
endpoint = urb->HcdUrbCloseEndpoint.HcdEndpoint;
ASSERT_ENDPOINT(endpoint);
} else {
//#if DBG
// UHCD_KdPrint((2, "'originalTransfer Buffer = 0x%x \n",
// urb->HcdUrbCommonTransfer.TransferBuffer));
// // Trash the TransferBuffer field - we only use MDLs
// urb->HcdUrbCommonTransfer.TransferBuffer = (PVOID)-1;
//#endif
// allocate some working space and attach it to
// this urb
endpoint = HCD_AREA(urb).HcdEndpoint;
ASSERT_ENDPOINT(endpoint);
if (urb->HcdUrbCommonTransfer.TransferBufferLength >
endpoint->MaxTransferSize) {
ntStatus = STATUS_INVALID_PARAMETER;
URB_HEADER(urb).Status = USBD_STATUS_INVALID_PARAMETER;
UHCD_CompleteIrp(DeviceObject, Irp, ntStatus, 0, NULL);
break;
}
siz = COMPUTE_HCD_EXTENSION_SIZE(urb);
HCD_AREA(urb).HcdExtension =
GETHEAP(NonPagedPool, siz);
if (HCD_AREA(urb).HcdExtension == NULL) {
ntStatus = STATUS_INSUFFICIENT_RESOURCES;
URB_HEADER(urb).Status = USBD_STATUS_NO_MEMORY;
UHCD_CompleteIrp(DeviceObject, Irp, ntStatus, 0, NULL);
break;
}
RtlZeroMemory(HCD_AREA(urb).HcdExtension, siz);
}
#ifdef ROOT_HUB
if (endpoint->EndpointFlags & EPFLAG_ROOT_HUB) {
// These routines will either complete
// the irp or mark it pending.
switch (URB_HEADER(urb).Function) {
case URB_FUNCTION_CONTROL_TRANSFER:
ntStatus = UHCD_RootHub_ControlTransfer(DeviceObject,
Irp,
urb);
// note: URB and IRP may be gone
break;
case URB_FUNCTION_BULK_OR_INTERRUPT_TRANSFER:
ntStatus = UHCD_RootHub_InterruptTransfer(DeviceObject,
Irp,
urb);
// note: URB and IRP may be gone
break;
case URB_FUNCTION_HCD_CLOSE_ENDPOINT:
ntStatus = UHCD_RootHub_CloseEndpoint(DeviceObject,
Irp,
urb);
break;
default:
//BUGBUG could just complete it with an error
//here
// this is probably a bug in the hub driver
UHCD_KdTrap(("Bogus transfer request to root hub\n"));
}
break;
}
#endif
URB_HEADER(urb).Status = UHCD_STATUS_PENDING_STARTIO;
ntStatus = STATUS_PENDING;
UHCD_KdPrint((2, "'Queue Irp To StartIo\n"));
Irp->IoStatus.Status = ntStatus;
Irp->IoStatus.Information = 0;
IoMarkIrpPending(Irp);
IoStartPacket(DeviceObject,
Irp,
0,
UHCD_StartIoCancel);
break;
case URB_FUNCTION_ISOCH_TRANSFER:
//
// validate max transfer size
// this will activate the rollover interrupts
UHCD_WakeIdle(DeviceObject);
endpoint = HCD_AREA(urb).HcdEndpoint;
ASSERT_ENDPOINT(endpoint);
// don't ref TDList
endpoint->TDList = (PVOID) -1;
if (urb->HcdUrbCommonTransfer.TransferBufferLength >
endpoint->MaxTransferSize) {
ntStatus = STATUS_INVALID_PARAMETER;
URB_HEADER(urb).Status = USBD_STATUS_INVALID_PARAMETER;
UHCD_CompleteIrp(DeviceObject, Irp, ntStatus, 0, NULL);
break;
}
// allocate some working space and attach it to
// this urb
siz = COMPUTE_HCD_EXTENSION_SIZE(urb);
HCD_AREA(urb).HcdExtension =
GETHEAP(NonPagedPool, siz);
if (HCD_AREA(urb).HcdExtension == NULL) {
ntStatus = STATUS_INSUFFICIENT_RESOURCES;
URB_HEADER(urb).Status = USBD_STATUS_NO_MEMORY;
UHCD_CompleteIrp(DeviceObject, Irp, ntStatus, 0, NULL);
break;
}
RtlZeroMemory(HCD_AREA(urb).HcdExtension, siz);
if (HCD_AREA(urb).HcdExtension == NULL) {
UHCD_KdBreak((2, "'Unable to allocate working space for isoch transfer\n"));
URB_HEADER(urb).Status = USBD_STATUS_NO_MEMORY;
UHCD_CompleteIrp(DeviceObject, Irp, ntStatus, 0, NULL);
break;
}
URB_HEADER(urb).Status = UHCD_STATUS_PENDING_STARTIO;
ntStatus = STATUS_PENDING;
UHCD_KdPrint((2, "'Queue Irp To StartIo\n"));
Irp->IoStatus.Status = ntStatus;
Irp->IoStatus.Information = 0;
IoMarkIrpPending(Irp);
#if DBG
{
UHCD_KdPrint((2, "'cf = %x",
UHCD_GetCurrentFrame(DeviceObject)));
}
#endif
if (endpoint->EndpointFlags & EPFLAG_FAST_ISO) {
KIRQL irql;
KeRaiseIrql(DISPATCH_LEVEL, &irql);
UHCD_Transfer_StartIo(DeviceObject, Irp);
KeLowerIrql(irql);
} else {
IoStartPacket(DeviceObject,
Irp,
0,
UHCD_StartIoCancel);
}
break;
case URB_FUNCTION_GET_CURRENT_FRAME_NUMBER:
// this will activate the rollover interrupts
UHCD_WakeIdle(DeviceObject);
urb->UrbGetCurrentFrameNumber.FrameNumber =
UHCD_GetCurrentFrame(DeviceObject);
LOGENTRY(LOG_MISC, 'gcfR',
Irp, urb, urb->UrbGetCurrentFrameNumber.FrameNumber);
URB_HEADER(urb).Status = USBD_STATUS_SUCCESS;
UHCD_CompleteIrp(DeviceObject, Irp, ntStatus, 0, NULL);
break;
case URB_FUNCTION_HCD_GET_ENDPOINT_STATE:
case URB_FUNCTION_HCD_SET_ENDPOINT_STATE:
URB_HEADER(urb).Status = UHCD_STATUS_PENDING_STARTIO;
ntStatus = STATUS_PENDING;
UHCD_KdPrint((2, "'Queue Irp To StartIo\n"));
Irp->IoStatus.Status = ntStatus;
Irp->IoStatus.Information = 0;
IoMarkIrpPending(Irp);
IoStartPacket(DeviceObject,
Irp,
0,
UHCD_StartIoCancel);
break;
case URB_FUNCTION_HCD_ABORT_ENDPOINT:
endpoint = urb->HcdUrbAbortEndpoint.HcdEndpoint;
ASSERT_ENDPOINT(endpoint);
//
// Mark the endpoint so that we abort all the transfers the
// next time we process it.
//
SET_EPFLAG(endpoint,
EPFLAG_ABORT_PENDING_TRANSFERS | EPFLAG_ABORT_ACTIVE_TRANSFERS);
URB_HEADER(urb).Status = UHCD_STATUS_PENDING_STARTIO;
ntStatus = STATUS_PENDING;
UHCD_KdPrint((2, "'Queue Irp To StartIo\n"));
Irp->IoStatus.Status = ntStatus;
Irp->IoStatus.Information = 0;
IoMarkIrpPending(Irp);
IoStartPacket(DeviceObject,
Irp,
0,
UHCD_StartIoCancel);
break;
case URB_FUNCTION_SET_FRAME_LENGTH:
{
CHAR sofModify;
//get the current value
sofModify = (CHAR) (READ_PORT_UCHAR(SOF_MODIFY_REG(deviceExtension)));
sofModify += (CHAR) urb->UrbSetFrameLength.FrameLengthDelta;
WRITE_PORT_UCHAR(SOF_MODIFY_REG(deviceExtension), (UCHAR) sofModify);
URB_HEADER(urb).Status = USBD_STATUS_SUCCESS;
UHCD_CompleteIrp(DeviceObject, Irp, ntStatus, 0, NULL);
UHCD_KdPrint((2, "'Irp Completed in dispatch\n"));
}
break;
case URB_FUNCTION_GET_FRAME_LENGTH:
{
CHAR sofModify;
//get the current value
sofModify = (CHAR) (READ_PORT_UCHAR(SOF_MODIFY_REG(deviceExtension)));
urb->UrbGetFrameLength.FrameNumber = UHCD_GetCurrentFrame(DeviceObject);
urb->UrbGetFrameLength.FrameLength = UHCD_12MHZ_SOF + sofModify;
URB_HEADER(urb).Status = USBD_STATUS_SUCCESS;
UHCD_CompleteIrp(DeviceObject, Irp, ntStatus, 0, NULL);
UHCD_KdPrint((2, "'Irp Completed in dispatch\n"));
}
break;
default:
UHCD_KdPrint((2, "'UHCD_URB_Dispatch -- invalid URB function (%x)\n",
URB_HEADER(urb).Function));
URB_HEADER(urb).Status = USBD_STATUS_INVALID_URB_FUNCTION;
ntStatus = STATUS_INVALID_PARAMETER;
UHCD_CompleteIrp(DeviceObject, Irp, ntStatus, 0, NULL);
}
UHCD_KdPrint((2, "'exit UHCD_URB_Dispatch (%x)\n", ntStatus));
return ntStatus;
}
VOID
UHCD_StartIo(
IN PDEVICE_OBJECT DeviceObject,
IN PIRP Irp
)
/*++
Routine Description:
Process URBs from the startIo routine.
Arguments:
DeviceObject - pointer to a device object
Irp - pointer to an I/O Request Packet
Return Value:
--*/
{
PHCD_URB urb;
KIRQL irql;
UHCD_KdPrint((2, "'enter UHCD_StartIo\n"));
//
// see if the Irp was canceled
//
urb = URB_FROM_IRP(Irp);
IoAcquireCancelSpinLock(&irql);
IoSetCancelRoutine(Irp, NULL);
if (Irp->Cancel) {
TEST_TRAP();
Irp->IoStatus.Status = STATUS_CANCELLED;
IoReleaseCancelSpinLock(irql);
IoCompleteRequest(Irp, IO_NO_INCREMENT);
IoStartNextPacket(DeviceObject, FALSE);
} else {
if (UHCD_IS_TRANSFER(urb)) {
IoSetCancelRoutine(Irp, UHCD_TransferCancel);
} else {
IoSetCancelRoutine(Irp, NULL);
}
IoReleaseCancelSpinLock(irql);
switch(URB_HEADER(urb).Function) {
case URB_FUNCTION_HCD_OPEN_ENDPOINT:
UHCD_OpenEndpoint_StartIo(DeviceObject, Irp);
break;
case URB_FUNCTION_HCD_CLOSE_ENDPOINT:
LOGENTRY(LOG_MISC, 'CLEP',
(PUHCD_ENDPOINT)urb->HcdUrbAbortEndpoint.HcdEndpoint, 0,
0);
UHCD_CloseEndpoint_StartIo(DeviceObject, Irp);
break;
case URB_FUNCTION_ISOCH_TRANSFER:
case URB_FUNCTION_CONTROL_TRANSFER:
case URB_FUNCTION_BULK_OR_INTERRUPT_TRANSFER:
UHCD_Transfer_StartIo(DeviceObject, Irp);
break;
case URB_FUNCTION_HCD_ABORT_ENDPOINT:
// walk the pending transfer list, since the
// enpoint is halted no new transfers will
// be dequeued by EndpointWorker and the only
// way to clear the HALT is thru
// GetSetEndpointState_StartIo so it is safe to
// mess with the list.
{
PIRP irp;
PHCD_URB urbtmp;
PUHCD_ENDPOINT endpoint;
PLIST_ENTRY listEntry;
KIRQL irql;
endpoint = (PUHCD_ENDPOINT) urb->HcdUrbAbortEndpoint.HcdEndpoint;
ASSERT_ENDPOINT(endpoint);
UHCD_KdPrint((2, "'Aborting endpoint %x flags = %x\n",
endpoint, endpoint->EndpointFlags));
LOGENTRY(LOG_MISC, 'ABRP', endpoint, 0, 0);
LOCK_ENDPOINT_PENDING_LIST(endpoint, irql, 'lck4');
while (!IsListEmpty(&endpoint->PendingTransferList)) {
listEntry = RemoveHeadList(&endpoint->PendingTransferList);
urbtmp = (PHCD_URB) CONTAINING_RECORD(listEntry,
struct _URB_HCD_COMMON_TRANSFER,
hca.HcdListEntry);
UNLOCK_ENDPOINT_PENDING_LIST(endpoint, irql, 'ulk4');
UHCD_KdPrint((2, "'Aborting urb = %x\n", urbtmp));
URB_HEADER(urbtmp).Status = USBD_STATUS_CANCELED;
//
// complete the request
//
irp = HCD_AREA(urbtmp).HcdIrp;
LOGENTRY(LOG_MISC, 'ABRc', endpoint, irp, 0);
UHCD_CompleteIrp(DeviceObject,
irp,
STATUS_CANCELLED,
0,
urbtmp);
LOCK_ENDPOINT_PENDING_LIST(endpoint, irql, 'lck5');
}
UNLOCK_ENDPOINT_PENDING_LIST(endpoint, irql, 'ulk5');
// do cleanup for fast iso
if (endpoint->EndpointFlags & EPFLAG_FAST_ISO) {
ULONG f;
for (f=0; f < FRAME_LIST_SIZE; f++) {
UHCD_CleanupFastIsoTD(DeviceObject,
endpoint,
f,
FALSE);
}
}
}
//
// we have already set the abortPendingTransfers and
// abortActiveTransfers flags in the dispatch
// routine. Now we need to request an interrupt
// so that any active transfers will be cleaned
// up.
//
UHCD_RequestInterrupt(DeviceObject, -2);
IoStartNextPacket(DeviceObject, FALSE);
URB_HEADER(urb).Status = USBD_STATUS_SUCCESS;
UHCD_CompleteIrp(DeviceObject, Irp, STATUS_SUCCESS, 0, NULL);
UHCD_KdPrint((2, "'UHCD Abort Endpoint Request Complete\n"));
break;
case URB_FUNCTION_HCD_GET_ENDPOINT_STATE:
case URB_FUNCTION_HCD_SET_ENDPOINT_STATE:
UHCD_GetSetEndpointState_StartIo(DeviceObject, Irp);
break;
default:
UHCD_KdPrint((2, "'UHCD_URB_StartIo -- invalid URB function (%x)\n", URB_HEADER(urb).Function));
URB_HEADER(urb).Status = USBD_STATUS_INVALID_URB_FUNCTION;
UHCD_CompleteIrp(DeviceObject, Irp, STATUS_SUCCESS, 0, NULL);
}
}
UHCD_KdPrint((2, "'exit UHCD_StartIo\n"));
}
VOID
UHCD_OpenEndpoint_StartIo(
IN PDEVICE_OBJECT DeviceObject,
IN PIRP Irp
)
/*++
Routine Description:
Create a UHCD endpoint, this function is called from UHCD_StartIo to
create a new endpoint structure.
Arguments:
DeviceObject - pointer to a device object
Irp - pointer to an I/O Request Packet
Return Value:
--*/
{
PHCD_URB urb;
PDEVICE_EXTENSION deviceExtension;
PUHCD_ENDPOINT endpoint;
PUSB_ENDPOINT_DESCRIPTOR endpointDescriptor;
KIRQL irql;
NTSTATUS ntStatus;
BOOLEAN haveBW;
ULONG offset;
UHCD_KdPrint((2, "'enter UHCD_OpenEndpoint_StartIo\n"));
deviceExtension = (PDEVICE_EXTENSION) DeviceObject->DeviceExtension;
//
// extract a pointer to the URB
//
urb = URB_FROM_IRP(Irp);
//
// make sure the length of the urb is what we expect
//
if (urb->HcdUrbOpenEndpoint.Length != sizeof(struct _URB_HCD_OPEN_ENDPOINT)) {
URB_HEADER(urb).Status = USBD_STATUS_INVALID_PARAMETER;
ntStatus = STATUS_INVALID_PARAMETER;
goto UHCD_OpenEndpoint_StartIo_Done;
}
//
// information about the endpoint comes from the USB endpoint
// descriptor passed in the URB.
//
endpointDescriptor = urb->HcdUrbOpenEndpoint.EndpointDescriptor;
URB_HEADER(urb).Status = USBD_STATUS_SUCCESS;
//
// Allocate resources for the endpoint, this includes an endpoint
// handle that will be passed to us in subsequent transfer requests
//
ntStatus = STATUS_INSUFFICIENT_RESOURCES;
endpoint = urb->HcdUrbOpenEndpoint.HcdEndpoint;
UHCD_ASSERT(endpoint != NULL);
ASSERT_ENDPOINT(endpoint);
if (endpoint) {
//
// initialize endpoint structures, state variables, queue head, ...
//
// bugbug -- may not need to call this for fastIiso
ntStatus = UHCD_FinishInitializeEndpoint(DeviceObject,
endpoint,
endpointDescriptor,
urb);
if (!NT_SUCCESS(ntStatus)) {
RETHEAP(endpoint);
goto UHCD_OpenEndpoint_StartIo_Done;
}
UHCD_KdPrint((2, "'Open Endpoint:\n"));
UHCD_KdPrint((2, "'Type = (%d) \n", endpoint->Type));
#if DBG
switch (endpoint->Type) {
case USB_ENDPOINT_TYPE_CONTROL:
UHCD_KdPrint((2, "'-control\n"));
break;
case USB_ENDPOINT_TYPE_ISOCHRONOUS:
UHCD_KdPrint((2, "'-iso\n"));
break;
case USB_ENDPOINT_TYPE_BULK:
UHCD_KdPrint((2, "'-bulk\n"));
break;
case USB_ENDPOINT_TYPE_INTERRUPT:
UHCD_KdPrint((2, "'-interrupt\n"));
}
if (USB_ENDPOINT_DIRECTION_IN(endpointDescriptor->bEndpointAddress)) {
UHCD_KdPrint((2, "'IN\n"));
} else {
UHCD_KdPrint((2, "'OUT\n"));
}
#endif
UHCD_KdPrint((2, "'EP Address = %d\n", endpoint->EndpointAddress));
UHCD_KdPrint((2, "'DEV Address = %d\n", endpoint->DeviceAddress));
UHCD_KdPrint((2, "'MaxPacket = %d\n", endpoint->MaxPacketSize));
UHCD_KdPrint((2, "'Interval: Requested = %d, Selected = %d\n", endpointDescriptor->bInterval,
endpoint->Interval));
//
// Now attempt to allocate the bandwidth we'll need to
// open this endpoint
//
if (endpoint->Type == USB_ENDPOINT_TYPE_INTERRUPT) {
ULONG i;
// check bw available in all locations and
// pick the least loaded frame
offset = 0;
for (i=0; i< endpoint->Interval; i++) {
if (deviceExtension->BwTable[i] > deviceExtension->BwTable[offset]) {
offset = i;
}
}
haveBW = UHCD_AllocateBandwidth(DeviceObject,
endpoint,
offset);
if (!haveBW) {
//
// could not use the least loaded frame, just grab the
// first frame we can fit in.
//
for (offset=0; offset<endpoint->Interval; offset++) {
haveBW = UHCD_AllocateBandwidth(DeviceObject,
endpoint,
offset);
if (haveBW) {
UHCD_KdPrint((2, "'using offset %d\n", offset));
break;
}
}
}
urb->HcdUrbOpenEndpoint.ScheduleOffset = offset;
} else {
offset = 0;
haveBW = UHCD_AllocateBandwidth(DeviceObject,
endpoint,
offset);
}
if (!haveBW) { // no offset
ULONG cnt;
//
// insufficient bandwidth to open this
// endpoint.
//
URB_HEADER(urb).Status = USBD_STATUS_NO_BANDWIDTH;
UHCD_KdPrint((0, "'warning: no bandwidth for endpoint\n"));
for (cnt=0; cnt< endpoint->MaxRequests; cnt++) {
UHCD_FreeHardwareDescriptors(DeviceObject,
endpoint->HardwareDescriptorList[cnt]);
}
RETHEAP(endpoint);
goto UHCD_OpenEndpoint_StartIo_Done;
}
if (endpoint->EndpointFlags & EPFLAG_FAST_ISO) {
goto UHCD_OpenEndpoint_StartIo_Done;
}
//
// put the endpoint in to our list of active endpoints
//
// BUGBUG may want to wait for the first transfer to
// do this.
//
LOCK_ENDPOINT_LIST(deviceExtension, irql);
if (deviceExtension->EndpointListBusy) {
// if the endpoint list is busy we have to put the endpoint
// on a lookaside list so that the ISRDPC can add it later
InsertHeadList(&deviceExtension->EndpointLookAsideList, &endpoint->ListEntry);
} else {
InsertHeadList(&deviceExtension->EndpointList, &endpoint->ListEntry);
}
UNLOCK_ENDPOINT_LIST(deviceExtension, irql);
//
// Put this Queue Head in the Schedule.
//
//
// This routine will insert the queue head in the proper place
// in the schedule and add the endpoint to the endpoint list
// so that completed transfers will be detected.
//
KeAcquireSpinLock(&deviceExtension->HcScheduleSpin, &irql);
UHCD_InsertQueueHeadInSchedule(DeviceObject,
endpoint,
endpoint->QueueHead,
offset); // no offset
// clear the idle flag just in case it got set
CLR_EPFLAG(endpoint, EPFLAG_IDLE);
KeReleaseSpinLock(&deviceExtension->HcScheduleSpin, irql);
//
// return the endpoint handle
//
urb->HcdUrbOpenEndpoint.HcdEndpoint = endpoint;
ntStatus = STATUS_SUCCESS;
} /* if endpoint */
//
// Complete the IRP, status is in the status field of the URB
//
UHCD_KdPrint((2, "'exit UHCD_OpenEndpoint_StartIo (URB STATUS = %x)\n", URB_HEADER(urb).Status ));
UHCD_OpenEndpoint_StartIo_Done:
#if DBG
//
// sanity check our buffer pools
//
// UHCD_BufferPoolCheck(DeviceObject);
#endif
IoStartNextPacket(DeviceObject, FALSE);
UHCD_CompleteIrp(DeviceObject, Irp, ntStatus, 0, NULL);
return;
}
VOID
UHCD_CloseEndpoint_StartIo(
IN PDEVICE_OBJECT DeviceObject,
IN PIRP Irp
)
/*++
Routine Description:
Free a UHCD endpoint, if there are any pending transfers for this
endpoint this routine should fail.
Arguments:
DeviceObject - pointer to a device object
Irp - pointer to an I/O Request Packet
Return Value:
--*/
{
PHCD_URB urb;
PUHCD_ENDPOINT endpoint;
BOOLEAN outstandingTransfers = FALSE;
PDEVICE_EXTENSION deviceExtension;
ULONG i;
KIRQL irql;
BOOLEAN removed;
UHCD_KdPrint((2, "'enter UHCD_CloseEndpoint_StartIo\n"));
deviceExtension = (PDEVICE_EXTENSION) DeviceObject->DeviceExtension;
urb = URB_FROM_IRP(Irp);
endpoint = urb->HcdUrbCloseEndpoint.HcdEndpoint;
ASSERT_ENDPOINT(endpoint);
if (endpoint->EndpointFlags & EPFLAG_FAST_ISO) {
UHCD_UnInitializeFastIsoEndpoint(DeviceObject,
endpoint);
UHCD_FreeBandwidth(DeviceObject, endpoint, endpoint->Offset);
}
//
// if there are any pending transfers fail this request
//
LOCK_ENDPOINT_PENDING_LIST(endpoint, irql, 'lck4');
//
// Do we have any transfers pending?
//
outstandingTransfers = !IsListEmpty(&endpoint->PendingTransferList);
for (i=0; !outstandingTransfers && i < endpoint->MaxRequests; i++) {
//
// no outstanding transfers in the queue, check the active list
// -- if some transfers get retired while we walk the list that
// is OK.
//
if (endpoint->ActiveTransfers[i] != NULL) {
outstandingTransfers = TRUE;
}
}
UNLOCK_ENDPOINT_PENDING_LIST(endpoint, irql, 'ulk4');
if (outstandingTransfers) {
//
// If we have outstanding transfers then we fail the
// request
//
URB_HEADER(urb).Status = USBD_STATUS_ERROR_BUSY;
} else {
//
// Remove the endpoint Queue from the schedule
//
KeAcquireSpinLock(&deviceExtension->HcScheduleSpin, &irql);
removed = UHCD_RemoveQueueHeadFromSchedule(DeviceObject,
endpoint,
endpoint->QueueHead,
TRUE);
KeReleaseSpinLock(&deviceExtension->HcScheduleSpin, irql);
// stop 'NoDMA' transfers
if (endpoint->EndpointFlags & EPFLAG_DBL_BUFFER) {
UHCD_StopNoDMATransfer(DeviceObject,
endpoint);
}
//
// Put the endpoint on a queue to be freed at after
// the next frame has executed
//
// At this point the hardware links have been updated to remove this
// endpoint.
//
// we note the frame when it is safe to retire the endpoint so that
// the queue head may be freed safely by the ISR DPC routine next time
// we take an interrupt.
endpoint->FrameToClose = UHCD_GetCurrentFrame(DeviceObject)+2;
// BUGBUG this needs to be protected from the ISR DPC
// routine where these things are actually freed.
// queue it to be released
if (removed) {
InsertTailList(&deviceExtension->ClosedEndpointList, &endpoint->ListEntry);
}
URB_HEADER(urb).Status = USBD_STATUS_SUCCESS;
}
IoStartNextPacket(DeviceObject, FALSE);
UHCD_CompleteIrp(DeviceObject, Irp, STATUS_SUCCESS, 0, NULL);
UHCD_KdPrint((2, "'exit UHCD_CloseEndpoint_StartIo\n"));
}
VOID
UHCD_InsertQueueHeadInSchedule(
IN PDEVICE_OBJECT DeviceObject,
IN PUHCD_ENDPOINT Endpoint,
IN PHW_QUEUE_HEAD QueueHead,
IN ULONG Offset
)
/*++
Routine Description:
Inserts an initialized queue head into the schedule
Queue head schedule looks like this:
PQH = persistant queue head
CQH = control Queue Head
IQH = Interrupt Queue Head
BQH = Bulk Queue Head
The persistant queue head software links
look like this:
|>---->---->-|
| <- <- | prev
BQH PQH CQH
| -> -> | next
|<----<----<-|
IQH->IQH->PQH
Hardware links look like this:
PQH->CQH->BQH->| (reclimaton on)
|-<-|
or
PQH->CQH->BQH->| (reclimation off)
|<-T-| (T BIT SET)
Iso/Interrupt hardware links:
ISO->IQH->PQH
Arguments:
DeviceObject - device object for this controller.
Endpoint - endpoint this Queue Head belongs to.
QueueHead - queue head to insert in schedule.
Return Value:
--*/
{
PDEVICE_EXTENSION deviceExtension;
PHW_QUEUE_HEAD persistantQueueHead, queueHeadForFrame,
nextQueueHead, prevQueueHead;
#ifdef RECLAIM_BW
PHW_QUEUE_HEAD firstBulkQueueHead;
#endif
ULONG i, interval;
BOOLEAN fixFrameList = FALSE;
UHCD_ASSERT(!(Endpoint->EndpointFlags & EPFLAG_ED_IN_SCHEDULE));
UHCD_KdPrint((2, "'enter InsertQueueHeadInSchedule\n"));
ASSERT_ENDPOINT(Endpoint);
ASSERT_QUEUE_HEAD(QueueHead);
deviceExtension = (PDEVICE_EXTENSION) DeviceObject->DeviceExtension;
persistantQueueHead = deviceExtension->PersistantQueueHead;
switch(Endpoint->Type) {
case USB_ENDPOINT_TYPE_CONTROL:
LOGENTRY(LOG_MISC, 'Cep+', Endpoint, 0, DeviceObject);
//
// before
//
// a b
// -PQH<->CQH-
// |---<->---|
//
// after
//
// a b
// -PQH<->CQH(new)<->CQH-
// |-------<->----------|
//
// insert in list
QueueHead->Next = persistantQueueHead->Next; //point to b
QueueHead->Prev = persistantQueueHead; //point to a
QueueHead->Next->Prev = QueueHead; // item b points back to new
// PQH points to the new element
QueueHead->Prev->Next = QueueHead; // item a points to new
// Fix up hardware links
QueueHead->HW_HLink = persistantQueueHead->HW_HLink;
// we are in the chain just after the Persistant Queue Head
// this line activates the QH
persistantQueueHead->HW_HLink =
QueueHead->PhysicalAddress;
break;
case USB_ENDPOINT_TYPE_INTERRUPT:
LOGENTRY(LOG_MISC, 'Iep+', Endpoint, QueueHead, DeviceObject);
QueueHead->Next = 0;
QueueHead->Prev = 0;
interval = Endpoint->Interval;
UHCD_ASSERT(Offset < interval);
UHCD_ASSERT(interval != 0);
LOGENTRY(LOG_MISC, 'Iep+',
Endpoint,
interval,
&deviceExtension->InterruptSchedule[0]);
// MAX_INTERVAL is the maximum polling interval we support in UHCD (power of 2).
// the requested polling intervals is always rounded to the next lowest power of 2.
// We keep an array (InterruptSchedule) of MAX_INTERVAL with the virtual addresses
// of the queue heads in the schedule. The size of InterruptSchedule is always a
// multiple of MAX_FRAME. i.e. InterruptSchedule contains the virtual addresses of
// queue heads for the first MAX_INTERVAL frames in the schedule.
UHCD_ASSERT(interval<=MAX_INTERVAL);
UHCD_ASSERT(Offset<interval);
UHCD_ASSERT(MAX_INTERVAL % interval == 0);
for (i=Offset; i<MAX_INTERVAL; i+=interval) {
// select the queue head
queueHeadForFrame = deviceExtension->InterruptSchedule[i];
LOGENTRY(LOG_MISC, 'Iqhf',
0, //queueHeadForFrame->Endpoint->Interval,
queueHeadForFrame,
i);
// find the appropriate place to add ourselves
if (queueHeadForFrame == persistantQueueHead ||
interval >= queueHeadForFrame->Endpoint->Interval) {
// if the first entry is the persistant queue head or our polling
// interval is greater or equal than this queue head then we just
// add in front of this one, we become the root node for this
// frame.
LOGENTRY(LOG_MISC, 'I>SC', 0, QueueHead, i);
deviceExtension->InterruptSchedule[i] = QueueHead;
fixFrameList = TRUE;
if (QueueHead->Next == 0) {
QueueHead->Next = queueHeadForFrame;
// update hardware link here
QueueHead->HW_HLink = queueHeadForFrame->PhysicalAddress;
} else {
// ounce the next pointer is updated
// it should never change
UHCD_ASSERT(QueueHead->Next == queueHeadForFrame);
}
} else {
// if our polling interval is less than the current queue
// head we need to insert in the proper position
LOGENTRY(LOG_MISC, 'I<SC', queueHeadForFrame, QueueHead, i);
nextQueueHead = queueHeadForFrame;
do {
prevQueueHead = nextQueueHead;
nextQueueHead = nextQueueHead->Next;
} while (nextQueueHead != persistantQueueHead &&
nextQueueHead->Endpoint->Interval > interval);
LOGENTRY(LOG_MISC, 'I<SQ', nextQueueHead,
QueueHead, prevQueueHead);
UHCD_ASSERT(nextQueueHead != 0);
// link in to the chain
if (QueueHead->Next == 0) {
QueueHead->Next = nextQueueHead;
// update hardware link here
QueueHead->HW_HLink = nextQueueHead->PhysicalAddress;
} else {
UHCD_ASSERT(QueueHead->Next == nextQueueHead ||
nextQueueHead == QueueHead);
}
prevQueueHead->Next = QueueHead;
// update hardware link here
prevQueueHead->HW_HLink = QueueHead->PhysicalAddress;
}
//
// repeat the process until we
// have visited all possible nodes for this queue head
//
}
// now update the physical frame list based on the virtual list
// if we have modified it.
if (fixFrameList) {
UHCD_CopyInterruptScheduleToFrameList(DeviceObject);
}
break;
case USB_ENDPOINT_TYPE_BULK:
//
// before
//
// b a
// -BQH<->PQH-
// |---<->---|
//
// after
//
// b a
// -BQH<->BQH(new)<->PQH-
// |-------<->----------|
//
// We need to add this endpoint to the tail of the
// persistant queue
//
LOGENTRY(LOG_MISC, 'Bep+', Endpoint, 0, DeviceObject);
// set up queue head fields
UHCD_KdPrint((2, "'bulk QH = %x, %x\n", QueueHead, Endpoint));
// insert in list
QueueHead->Next = persistantQueueHead; // point to a
QueueHead->Prev = persistantQueueHead->Prev; // point to b
// first item points back to us
QueueHead->Prev->Next = QueueHead; // item b points to new
// head points to the element
// select the old tail element
prevQueueHead = persistantQueueHead->Prev; // remember b
persistantQueueHead->Prev = QueueHead; // item a points to new
#ifdef RECLAIM_BW
//
// Fix up hardware links
//
// BUGBUG
// NOTE: we are only reclaiming bandwidth for bulk right
// now.
//
if (deviceExtension->SteppingVersion == UHCD_B0_STEP) {
//TEST_TRAP();
//
// BW reclimation, point back to the first bulk queue head
// with no T bit set
//
// walk the list and find the first bulk
// queue head
firstBulkQueueHead = persistantQueueHead;
do {
PUHCD_ENDPOINT endpoint;
endpoint = firstBulkQueueHead->Endpoint;
if (endpoint &&
endpoint->Type == USB_ENDPOINT_TYPE_BULK) {
break;
}
firstBulkQueueHead = firstBulkQueueHead->Next;
} while (firstBulkQueueHead != persistantQueueHead);
UHCD_ASSERT(firstBulkQueueHead != persistantQueueHead);
QueueHead->HW_HLink = firstBulkQueueHead->PhysicalAddress;
deviceExtension->HcFlags |= HCFLAG_BWRECLIMATION_ENABLED;
} else {
// Fix up hardware links
// points to the control queue head with T bit set
QueueHead->HW_HLink = prevQueueHead->HW_HLink;
UHCD_ASSERT(QueueHead->HW_HLink & UHCD_CF_TERMINATE);
}
#else
// Fix up hardware links
// points to the control queue head with T bit set
QueueHead->HW_HLink = prevQueueHead->HW_HLink;
UHCD_ASSERT(QueueHead->HW_HLink & UHCD_CF_TERMINATE);
#endif
// we are in the chain just before the PersistantQueueHead
// this line activates the QH
prevQueueHead->HW_HLink =
QueueHead->PhysicalAddress;
break;
case USB_ENDPOINT_TYPE_ISOCHRONOUS:
LOGENTRY(LOG_MISC, 'Sep+', Endpoint, 0, DeviceObject);
break;
default:
// invalid endpoint type, probably a bug
UHCD_KdTrap(
("UHCD_InsertQueueHeadInSchedule inavlid endpoint type\n"));
}
SET_EPFLAG(Endpoint, EPFLAG_ED_IN_SCHEDULE);
UHCD_KdPrint((2, "'exit InsertQueueHeadInSchedule\n"));
return;
}
BOOLEAN
UHCD_RemoveQueueHeadFromSchedule(
IN PDEVICE_OBJECT DeviceObject,
IN PUHCD_ENDPOINT Endpoint,
IN PHW_QUEUE_HEAD QueueHead,
IN BOOLEAN RemoveFromEPList
)
/*++
Routine Description:
Removes a queue head from the schedule
Arguments:
DeviceObject - device object for this controller.
Endpoint - endpoint this Queue Head belongs to.
QueueHead - queue head to remove.
Return Value:
returns TRUE if EP was removed from ep list
--*/
{
PDEVICE_EXTENSION deviceExtension;
PHW_QUEUE_HEAD persistantQueueHead, queueHeadForFrame,
nextQueueHead, prevQueueHead;
BOOLEAN fixFrameList = FALSE;
ULONG i;
HW_DESCRIPTOR_PHYSICAL_ADDRESS physicalAddress;
KIRQL irql;
BOOLEAN removed = FALSE;
UHCD_KdPrint((2, "'enter RemoveQueueHeadFromSchedule\n"));
ASSERT_ENDPOINT(Endpoint);
if (Endpoint->EndpointFlags & EPFLAG_FAST_ISO) {
// nothing to remove
return FALSE;
}
ASSERT_QUEUE_HEAD(QueueHead);
deviceExtension = (PDEVICE_EXTENSION) DeviceObject->DeviceExtension;
persistantQueueHead = deviceExtension->PersistantQueueHead;
//
// Remove the Queue Head from the endpoint list
//
if (RemoveFromEPList) {
LOCK_ENDPOINT_LIST(deviceExtension, irql);
if (deviceExtension->EndpointListBusy) {
// mark the entry so we can remove it later
SET_EPFLAG(Endpoint, EPFLAG_EP_CLOSED);
} else {
RemoveEntryList(&Endpoint->ListEntry);
removed = TRUE;
}
UNLOCK_ENDPOINT_LIST(deviceExtension, irql);
}
if (!(Endpoint->EndpointFlags & EPFLAG_ED_IN_SCHEDULE)) {
LOGENTRY(LOG_MISC, 'NISc', Endpoint, 0, DeviceObject);
return removed;
}
switch(Endpoint->Type) {
case USB_ENDPOINT_TYPE_CONTROL:
case USB_ENDPOINT_TYPE_BULK:
// set up queue head fields
prevQueueHead = QueueHead->Prev;
nextQueueHead = QueueHead->Next;
// unlink software links
prevQueueHead->Next = QueueHead->Next;
nextQueueHead->Prev = QueueHead->Prev;
if ((QueueHead->HW_HLink & UHCD_DESCRIPTOR_PTR_MASK)
==
(QueueHead->PhysicalAddress & UHCD_DESCRIPTOR_PTR_MASK)) {
//
// Queue head points to itself, this means it is
// the only bulk queue in the list.
//
// This will only happen if we have BW reclaimation
// is enabled.
//
physicalAddress =
deviceExtension->PersistantQueueHead->PhysicalAddress;
SET_T_BIT(physicalAddress);
UHCD_ASSERT(physicalAddress & UHCD_CF_QUEUE);
prevQueueHead->HW_HLink = physicalAddress;
} else {
// Fix up hardware link
prevQueueHead->HW_HLink = QueueHead->HW_HLink;
}
break;
case USB_ENDPOINT_TYPE_INTERRUPT:
//
// Brute force method:
// Walk every frame in the InterruptSchedule and update
// any node that references this queue head.
//
for (i=0; i<MAX_INTERVAL; i++) {
queueHeadForFrame = deviceExtension->InterruptSchedule[i];
if (queueHeadForFrame == QueueHead) {
// Queue Head was root node for this frame
deviceExtension->InterruptSchedule[i] = QueueHead->Next;
fixFrameList = TRUE;
} else {
while (queueHeadForFrame != persistantQueueHead &&
queueHeadForFrame->Endpoint->Interval >=
QueueHead->Endpoint->Interval) {
if (queueHeadForFrame->Next == QueueHead) {
// found a link to our queue head,
// remove it
queueHeadForFrame->Next = QueueHead->Next;
// unlink from Hardware Queue
queueHeadForFrame->HW_HLink = QueueHead->HW_HLink;
}
queueHeadForFrame = queueHeadForFrame->Next;
}
}
if (fixFrameList) {
UHCD_CopyInterruptScheduleToFrameList(DeviceObject);
}
}
break;
case USB_ENDPOINT_TYPE_ISOCHRONOUS:
LOGENTRY(LOG_MISC, 'Sep-', Endpoint, 0, DeviceObject);
break;
default:
// Invalid endpoint type, probably a bug
UHCD_KdTrap(
("UHCD_RemoveQueueHeadFromSchedule inavlid endpoint type\n"));
}
CLR_EPFLAG(Endpoint,
EPFLAG_ED_IN_SCHEDULE);
UHCD_KdPrint((2, "'exit RemoveQueueHeadFromSchedule\n"));
return removed;
}
VOID
UHCD_CopyInterruptScheduleToFrameList(
IN PDEVICE_OBJECT DeviceObject
)
/*++
Routine Description:
Transfers the virtual interrupt schedule to the frame list
Arguments:
DeviceObject - device object for this controller.
Return Value:
None.
--*/
{
PDEVICE_EXTENSION deviceExtension;
ULONG i;
deviceExtension = (PDEVICE_EXTENSION) DeviceObject->DeviceExtension;
for (i=0; i < FRAME_LIST_SIZE; i++) {
if (i == 0 || i == FRAME_LIST_SIZE-1)
deviceExtension->TriggerTDList->TDs[i == 0 ? 0 : 1].HW_Link =
deviceExtension->InterruptSchedule[i % MAX_INTERVAL]->PhysicalAddress;
else {
ULONG currentTdCopy, currentTd;
PUHCD_ENDPOINT endpoint;
currentTd =
*( ((PULONG) (deviceExtension->FrameListVirtualAddress)+i));
currentTdCopy =
*( ((PULONG) (deviceExtension->FrameListCopyVirtualAddress)+i));
endpoint = UHCD_GetLastFastIsoEndpoint(DeviceObject);
// have fast iso?
if (endpoint) {
PFAST_ISO_DATA fastIsoData;
PHW_TRANSFER_DESCRIPTOR transferDescriptor;
// fast iso TDs are present, we will need to insert
// the interrupt schedule after these TDs.
fastIsoData = &endpoint->FastIsoData;
transferDescriptor = (PHW_TRANSFER_DESCRIPTOR)
(fastIsoData->IsoTDListVa + (i*32));
transferDescriptor->HW_Link =
deviceExtension->InterruptSchedule[i % MAX_INTERVAL]->PhysicalAddress;
} else {
// no fast iso -- just update the schedule
*( ((PULONG) (deviceExtension->FrameListCopyVirtualAddress)+i) ) =
deviceExtension->InterruptSchedule[i % MAX_INTERVAL]->PhysicalAddress;
// if the currentTd == the copy then we don't have any iso
// tds in the schedule so it is safe to update the schedule directly
if (currentTd == currentTdCopy) {
*( ((PULONG) (deviceExtension->FrameListVirtualAddress)+i) ) =
deviceExtension->InterruptSchedule[i % MAX_INTERVAL]->PhysicalAddress;
}
}
}
}
}
VOID
UHCD_StartIoCancel(
IN PDEVICE_OBJECT DeviceObject,
IN PIRP Irp
)
/*++
Routine Description:
Arguments:
Return Value:
--*/
{
PHCD_URB urb;
//
// Irp has not been processed by StartIo yet
//
LOGENTRY(LOG_MISC, 'sioC', Irp, 0, DeviceObject);
if (DeviceObject->CurrentIrp == Irp) {
LOGENTRY(LOG_MISC, 'curI', Irp, 0, DeviceObject);
IoReleaseCancelSpinLock(Irp->CancelIrql);
} else {
LOGENTRY(LOG_MISC, 'Ncur', Irp, 0, DeviceObject);
if (KeRemoveEntryDeviceQueue(&DeviceObject->DeviceQueue,
&Irp->Tail.Overlay.DeviceQueueEntry)) {
LOGENTRY(LOG_MISC, 'YDVQ', Irp, 0, DeviceObject);
TEST_TRAP();
urb = (PHCD_URB) URB_FROM_IRP(Irp);
IoReleaseCancelSpinLock(Irp->CancelIrql);
while (urb) {
URB_HEADER(urb).Status = USBD_STATUS_CANCELED;
if (UHCD_IS_TRANSFER(urb)) {
urb = urb->HcdUrbCommonTransfer.UrbLink;
} else {
break;
}
}
Irp->IoStatus.Status = STATUS_CANCELLED;
Irp->IoStatus.Information = 0;
USBD_CompleteRequest(Irp,
IO_NO_INCREMENT);
} else {
LOGENTRY(LOG_MISC, 'NDVQ', Irp, 0, DeviceObject);
TEST_TRAP();
IoReleaseCancelSpinLock(Irp->CancelIrql);
}
}
}
VOID
UHCD_GetSetEndpointState_StartIo(
IN PDEVICE_OBJECT DeviceObject,
IN PIRP Irp
)
/*++
Routine Description:
Change or report state of an endpoint
Arguments:
DeviceObject - pointer to a device object
Irp - pointer to an I/O Request Packet
Return Value:
--*/
{
PUHCD_ENDPOINT endpoint;
PHCD_URB urb;
BOOLEAN outstandingTransfers;
ULONG i;
KIRQL irql;
urb = (PHCD_URB) URB_FROM_IRP(Irp);
endpoint = (PUHCD_ENDPOINT) urb->HcdUrbEndpointState.HcdEndpoint;
ASSERT_ENDPOINT(endpoint);
//
// Do we have any transfers pending?
//
irql = KeGetCurrentIrql();
LOCK_ENDPOINT_PENDING_LIST(endpoint, irql, 'lck5');
outstandingTransfers = !IsListEmpty(&endpoint->PendingTransferList);
#if DBG
if (outstandingTransfers) {
UHCD_KdPrint((2, "'GET_ENDPOINT_STATE ep has pending transfers\n"));
}
#endif
for (i=0; !outstandingTransfers && i < endpoint->MaxRequests; i++) {
//
// no outstanding transfers in the queue, check the active list
// -- if some transfers get retired while we walk the list that
// is OK.
//
if (endpoint->ActiveTransfers[i] != NULL) {
UHCD_KdPrint((2, "'GETSET_ENDPOINT_STATE ep has active transfers\n"));
outstandingTransfers = TRUE;
}
}
UNLOCK_ENDPOINT_PENDING_LIST(endpoint, irql, 'ulk5');
switch (urb->HcdUrbEndpointState.Function) {
case URB_FUNCTION_HCD_GET_ENDPOINT_STATE:
urb->HcdUrbEndpointState.HcdEndpointState = 0;
if (endpoint->EndpointFlags & EPFLAG_HOST_HALTED) {
UHCD_KdPrint((2, "'GET_ENDPOINT_STATE host halted\n"));
urb->HcdUrbEndpointState.HcdEndpointState |=
HCD_ENDPOINT_HALTED;
}
if (outstandingTransfers) {
urb->HcdUrbEndpointState.HcdEndpointState |=
HCD_ENDPOINT_TRANSFERS_QUEUED;
}
URB_HEADER(urb).Status = USBD_STATUS_SUCCESS;
UHCD_KdPrint((2, "'GET_ENDPOINT_STATE state = %x\n",
urb->HcdUrbEndpointState.HcdEndpointState));
break;
case URB_FUNCTION_HCD_SET_ENDPOINT_STATE:
if (!outstandingTransfers) {
LOGENTRY(LOG_MISC, 'cla2', endpoint, 0, 0);
CLR_EPFLAG(endpoint,
EPFLAG_ABORT_PENDING_TRANSFERS |
EPFLAG_ABORT_ACTIVE_TRANSFERS);
}
UHCD_KdPrint((2, "'Set Enpoint State flags = %x\n", endpoint->EndpointFlags));
if (endpoint->EndpointFlags & (EPFLAG_ABORT_ACTIVE_TRANSFERS |
EPFLAG_ABORT_PENDING_TRANSFERS)) {
//fail the request
IoStartNextPacket(DeviceObject, FALSE);
// let USBD map the error for us
URB_HEADER(urb).Status = USBD_STATUS_ERROR_BUSY;
UHCD_CompleteIrp(DeviceObject, Irp, STATUS_SUCCESS, 0, NULL);
return;
}
if (endpoint->EndpointFlags & EPFLAG_DBL_BUFFER) {
// stop streaming data
UHCD_StopNoDMATransfer(DeviceObject,
endpoint);
}
//
// restore virgin status to the pipe
//
SET_EPFLAG(endpoint, EPFLAG_VIRGIN);
//
// set the data toggle back to 0
//
if (urb->HcdUrbEndpointState.HcdEndpointState &
HCD_ENDPOINT_RESET_DATA_TOGGLE) {
endpoint->DataToggle = 0;
}
if (!(urb->HcdUrbEndpointState.HcdEndpointState & HCD_ENDPOINT_HALTED)) {
//
// halt bit cleared, reset the endpoint.
//
LOGENTRY(LOG_MISC, 'cla3', endpoint, 0, 0);
CLR_EPFLAG(endpoint,
EPFLAG_ABORT_PENDING_TRANSFERS |
EPFLAG_ABORT_ACTIVE_TRANSFERS |
EPFLAG_HOST_HALTED);
//
// Start any transfers in the pending queue
//
if (endpoint->EndpointFlags & EPFLAG_DBL_BUFFER) {
// transfers will re-start on the next interrupt
UHCD_RequestInterrupt(DeviceObject, -2);
} else {
UHCD_EndpointWorker(DeviceObject, endpoint);
}
}
URB_HEADER(urb).Status = USBD_STATUS_SUCCESS;
break;
default:
// unknown command, probably a bug
UHCD_KdTrap(("Bogus Endpoint state command\n"));
}
IoStartNextPacket(DeviceObject, FALSE);
UHCD_CompleteIrp(DeviceObject, Irp, STATUS_SUCCESS, 0, NULL);
}
NTSTATUS
UHCD_FinishInitializeEndpoint(
IN PDEVICE_OBJECT DeviceObject,
IN PUHCD_ENDPOINT Endpoint,
IN PUSB_ENDPOINT_DESCRIPTOR EndpointDescriptor,
IN PHCD_URB Urb
)
/*++
Routine Description:
Change or report state of an endpoint
Arguments:
Endpoint - endpoint structure to initilaize
EndpointDescriptor - pointer to the USB endpoint descriptor
for this endpoint.
Urb - urb associated with the open request
Return Value:
--*/
{
UCHAR tmp;
ULONG i;
NTSTATUS ntStatus = STATUS_SUCCESS;
// note that some fields are already initilaized
ASSERT_ENDPOINT(Endpoint);
Endpoint->Type = USB_ENDPOINT_TYPE_MASK & EndpointDescriptor->bmAttributes;
Endpoint->EndpointAddress = EndpointDescriptor->bEndpointAddress;
Endpoint->MaxPacketSize = EndpointDescriptor->wMaxPacketSize;
Endpoint->DeviceAddress = (UCHAR) Urb->HcdUrbOpenEndpoint.DeviceAddress;
Endpoint->LastPacketDataToggle =
Endpoint->DataToggle = 0;
Endpoint->Interval = 0;
Endpoint->IdleTime = 0;
SET_EPFLAG(Endpoint, EPFLAG_VIRGIN);
SET_EPFLAG(Endpoint, EPFLAG_INIT);
Endpoint->MaxTransferSize = Urb->HcdUrbOpenEndpoint.MaxTransferSize;
// No DMA endpoint ?
#if DBG
if (Endpoint->EndpointFlags & EPFLAG_DBL_BUFFER) {
// client idicates that transfers will be mostly short
// in this case we will turn off short packet detect
// and double buffer all transfers to reduce the overhead of
// having to program each transfer to the hardware separately
UHCD_KdPrint((1, "'Client requesting double buffering EP = %x\n", Endpoint));
}
#endif
#if DBG
if (!(Endpoint->EndpointFlags & EPFLAG_ROOT_HUB)) {
UHCD_ASSERT(Endpoint->TDCount == UHCD_GetNumTDsPerEndoint(Endpoint->Type));
}
#endif
if (Urb->HcdUrbOpenEndpoint.HcdEndpointFlags & USBD_EP_FLAG_LOWSPEED) {
SET_EPFLAG(Endpoint, EPFLAG_LOWSPEED);
}
if (Urb->HcdUrbOpenEndpoint.HcdEndpointFlags & USBD_EP_FLAG_NEVERHALT) {
SET_EPFLAG(Endpoint, EPFLAG_NO_HALT);
}
KeInitializeSpinLock(&Endpoint->ActiveListSpin);
KeInitializeSpinLock(&Endpoint->PendingListSpin);
#if DBG
Endpoint->AccessPendingList = 0;
Endpoint->AccessActiveList = 0;
#endif
UHCD_KdPrint((2, "'MaxRequests = %d\n", Endpoint->MaxRequests));
UHCD_ASSERT(Endpoint->MaxRequests == MAX_REQUESTS(EndpointDescriptor,
Endpoint->EndpointFlags));
// Select the highest interval we support <= the requested interval.
// note: an interval of zero gets a oeriod of MAX_INTERVAL
tmp = EndpointDescriptor->bInterval;
Endpoint->Interval = MAX_INTERVAL;
if (EndpointDescriptor->bInterval > MAX_INTERVAL ||
EndpointDescriptor->bInterval == 0) {
tmp |= MAX_INTERVAL;
}
while ((Endpoint->Interval & tmp) == 0) {
Endpoint->Interval >>= 1;
}
//
// make sure that iso endpoints have an
// interval of 1
//
if (Endpoint->Type == USB_ENDPOINT_TYPE_ISOCHRONOUS) {
Endpoint->Interval = 1; //iso endpoints have a period of 1
}
InitializeListHead(&Endpoint->PendingTransferList);
for (i=0; i<UHCD_MAX_ACTIVE_TRANSFERS; i++) {
Endpoint->ActiveTransfers[i] = NULL;
}
return ntStatus;
}
USHORT
UHCD_GetNumTDsPerEndoint(
IN UCHAR EndpointType
)
/*++
Routine Description:
Return the number of TDs to use for this endpoint based on type
Arguments:
Return Value:
--*/
{
// use max TDs always for bulk to get max thru-put regardless
// of packet size.
// Historical Note:
// this is a change from Win98gold and Win98se,
// originally we only enabled this if MAX_PACKET was 64 to save
// memory for slower devices -- but vendors bitched about it so
// now we enable it regardless of packet size.
if (EndpointType == USB_ENDPOINT_TYPE_ISOCHRONOUS ||
EndpointType == USB_ENDPOINT_TYPE_BULK) {
return MAX_TDS_PER_ENDPOINT;
} else {
return MIN_TDS_PER_ENDPOINT;
}
}
VOID
UHCD_BW_Reclimation(
IN PDEVICE_OBJECT DeviceObject,
IN BOOLEAN Enable
)
/*++
Routine Description:
Turn on/off BW reclimation for the Bulk Queues
Arguments:
Return Value:
None.
--*/
{
PDEVICE_EXTENSION deviceExtension;
PHW_QUEUE_HEAD persistantQueueHead, firstBulkQueueHead,
lastBulkQueueHead;
deviceExtension = (PDEVICE_EXTENSION) DeviceObject->DeviceExtension;
persistantQueueHead = deviceExtension->PersistantQueueHead;
if ((deviceExtension->HcFlags & HCFLAG_BWRECLIMATION_ENABLED)
== Enable) {
// no state change just return;
return;
}
LOGENTRY(LOG_MISC, 'BRCL', deviceExtension, Enable, DeviceObject);
//
// BW reclimation, point back to the first bulk queue head
// with no T bit set
//
// walk the list and find the first bulk
// queue head
firstBulkQueueHead = persistantQueueHead;
do {
PUHCD_ENDPOINT endpoint;
endpoint = firstBulkQueueHead->Endpoint;
if (endpoint &&
endpoint->Type == USB_ENDPOINT_TYPE_BULK) {
break;
}
firstBulkQueueHead = firstBulkQueueHead->Next;
} while (firstBulkQueueHead != persistantQueueHead);
if (firstBulkQueueHead != persistantQueueHead) {
// no bulk endpoints
PHW_QUEUE_HEAD next;
PUHCD_ENDPOINT endpoint;
lastBulkQueueHead = firstBulkQueueHead;
do {
next = lastBulkQueueHead->Next;
endpoint = next->Endpoint;
if (endpoint == NULL ||
endpoint->Type != USB_ENDPOINT_TYPE_BULK) {
break;
}
lastBulkQueueHead = next;
} while (1);
if (Enable) {
//clear the T-bit to enable reclimation
LOGENTRY(LOG_MISC, 'BRC+', lastBulkQueueHead, 0, DeviceObject);
CLEAR_T_BIT(lastBulkQueueHead->HW_HLink);
deviceExtension->HcFlags |= HCFLAG_BWRECLIMATION_ENABLED;
} else {
//set the T-bit to disable reclimation
LOGENTRY(LOG_MISC, 'BRC-', lastBulkQueueHead, 0, DeviceObject);
SET_T_BIT(lastBulkQueueHead->HW_HLink);
deviceExtension->HcFlags &= ~HCFLAG_BWRECLIMATION_ENABLED;
}
}
}