/*++ Copyright (c) 2000 Microsoft Corporation Module Name: xid.cpp Abstract: Basic entry point implementation of the XID driver. Environment: Designed for XBOX. Notes: Revision History: 02-21-00 created by Mitchell Dernis (mitchd) --*/ #define XID_IMPLEMENTATION // // Pull in OS headers // #define _XAPI_ extern "C" { #include } #include #include #include #include // // Setup the debug information for this file (see ..\inc\debug.h) // #define MODULE_POOL_TAG '_DIX' #include DEFINE_USB_DEBUG_FUNCTIONS("XID"); // // Pull in public usb headers // #include // // Pull in xid headers // #include "xid.h" //------------------------------------------------------------------------------ // Declare XID's global variables. //------------------------------------------------------------------------------ XID_GLOBALS XID_Globals; //------------------------------------------------------------------------------ // Forward declaration of locally defined functions //------------------------------------------------------------------------------ VOID XID_WatchdogTimerProc( PKDPC, PVOID, PVOID, PVOID ); VOID XID_EnumStage1( PURB Urb, PXID_DEVICE_NODE XidNode ); VOID FASTCALL XID_fRemoveDeviceComplete( IN PXID_DEVICE_NODE XidNode ); PXID_DEVICE_NODE FASTCALL XID_fFindNode( IN PXID_TYPE_INFORMATION TypeInformation, IN ULONG Port ); USBD_STATUS FASTCALL XID_fOpenEndpoints( PXID_OPEN_DEVICE OpenDevice, PXINPUT_POLLING_PARAMETERS PollingParameters ); VOID FASTCALL XID_fCloseEndpoints( PXID_OPEN_DEVICE OpenDevice ); VOID XID_CloseEndpointStateMachine( PURB_CLOSE_ENDPOINT CloseUrb, PXID_OPEN_DEVICE OpenDevice ); VOID XID_NewInterruptData( PURB Urb, PXID_OPEN_DEVICE OpenDevice ); VOID XID_ClearInputStallComplete( PURB Urb, PXID_OPEN_DEVICE OpenDevice ); VOID XID_OutputComplete( PURB Urb, PXINPUT_FEEDBACK_INTERNAL OutputReport ); VOID FASTCALL XID_fOutputComplete1( PURB Urb, PXINPUT_FEEDBACK_INTERNAL OutputReport ); VOID XID_ClearOutputStallComplete( PURB Urb, PXINPUT_FEEDBACK_INTERNAL OutputReport ); VOID XID_EnumLegacy1(PURB Urb, PXID_DEVICE_NODE XidNode); //VOID XID_EnumLegacy2(PURB Urb, PXID_DEVICE_NODE XidNode); VOID XID_EnumKeyboard(PURB Urb, PXID_DEVICE_NODE XidNode); VOID XID_EnumKeyboardComplete(PURB Urb, PXID_DEVICE_NODE XidNode); PXID_KEYBOARD_SERVICES XID_pKeyboardServices = NULL; //------------------------------------------------------------------------------ // Declare XID types and class. //------------------------------------------------------------------------------ // Normally you would use macros for the device type table. // However, our types are dynamic, so we just want no types, // during init, we can enumerate our types and then update the tables. // USBD won't touch this entry until the first device is AddComplete. #define XID_Types NULL USB_CLASS_DRIVER_DECLARATION(XID_, XBOX_DEVICE_CLASS_INPUT_DEVICE, 0xFF, 0xFF) USB_CLASS_DRIVER_DECLARATION_DUPLICATE(XID_, 1, USB_DEVICE_CLASS_HUMAN_INTERFACE, 0xFF, 0xFF) #undef XID_Types #pragma data_seg(".XPP$ClassXID") USB_CLASS_DECLARATION_POINTER(XID_) USB_CLASS_DECLARATION_POINTER_DUPLICATE(XID_, 1) #pragma data_seg(".XPP$Data") USB_RESOURCE_REQUIREMENTS XID_gResourceRequirements = {USB_CONNECTOR_TYPE_DIRECT, 0, 0, 1, 0, 2, 10, 0, 0, 0}; #define DEVICE_TYPE_STACK_SIZE 20 #pragma code_seg(".XPPCINIT") EXTERNUSB VOID XID_Init(IUsbInit *UsbInit) /*++ Routine Description: Called at boot. Registers with usbpnp.sys. Intializes globals. --*/ { ULONG handleCount = 0; ULONG index; PXPP_DEVICE_TYPE devicesTypes[DEVICE_TYPE_STACK_SIZE]; BOOL fUseDefaultCount; PXID_TYPE_INFORMATION *ppTypeInformation; DWORD dwTypeIndex = 0; // // Verify that XOUTPUT_SIZE_OF_INTERNAL_HEADER is defined correctly // in input.h // C_ASSERT(sizeof(XINPUT_FEEDBACK_HEADER_INTERNAL) == XINPUT_FEEDBACK_HEADER_INTERNAL_SIZE); //Sentry to protect against double init. This happens because we support XID and HID //class as the interface class. This support is here to stay. static BOOL fInitialized = FALSE; if(fInitialized) return; fInitialized = TRUE; // // Figure out how many nodes we need // XID_Globals.DeviceNodeCount = XGetPortCount()*XID_MAX_DEVICE_PER_PORT; // // Walk the Device Type Table (and properly register each type) // fUseDefaultCount = UsbInit->UseDefaultCount(); ppTypeInformation = (PXID_TYPE_INFORMATION *)((&XID_BeginTypeDescriptionTable)+1); while( (ULONG_PTR)ppTypeInformation < (ULONG_PTR)&XID_EndTypeDescriptionTable ) { if(*ppTypeInformation) { // Recorder the XPP type in the temporary table. ASSERT(dwTypeIndex < DEVICE_TYPE_STACK_SIZE); devicesTypes[dwTypeIndex++] = (*ppTypeInformation)->XppType; // Count how many handles we need, and record the user's // XInitDevices choice for count if(!fUseDefaultCount) { (*ppTypeInformation)->bRemainingHandles = UsbInit->GetMaxDeviceTypeCount((*ppTypeInformation)->XppType); RIP_ON_NOT_TRUE_WITH_MESSAGE( (*ppTypeInformation)->bRemainingHandles <= XGetPortCount(), "XInitDevices: requested more XDEVICE_TYPE_GAMEPAD than available ports."); } handleCount += (*ppTypeInformation)->bRemainingHandles; } ppTypeInformation++; } // // In any case, we don't need more handles than we have device nodes. // if(handleCount > XID_Globals.DeviceNodeCount) { handleCount = XID_Globals.DeviceNodeCount; } // // Allocate a block for the device type table, device nodes and handles // ULONG allocSize = sizeof(PXPP_DEVICE_TYPE)*dwTypeIndex + sizeof(XID_DEVICE_NODE)*XID_Globals.DeviceNodeCount + sizeof(XID_OPEN_DEVICE)*handleCount; ULONG_PTR memory = (ULONG_PTR) RTL_ALLOCATE_HEAP(allocSize); ASSERT(memory); // // Fix our device type table // XID_Description.DeviceTypeCount = dwTypeIndex; XID_Description.DeviceTypes = (PXPP_DEVICE_TYPE *)memory; XID_1Description.DeviceTypeCount = dwTypeIndex; //The duplicate entry, for HID XID_1Description.DeviceTypes = (PXPP_DEVICE_TYPE *)memory; memory += (sizeof(PXPP_DEVICE_TYPE)*dwTypeIndex); memcpy(XID_Description.DeviceTypes, devicesTypes, sizeof(PXPP_DEVICE_TYPE)*dwTypeIndex); // // Create a free list of handles // XID_Globals.DeviceHandles = NULL; PXID_OPEN_DEVICE openDevices = (PXID_OPEN_DEVICE) memory; for(index=0; index < handleCount; index++) { XID_FreeHandle(openDevices++); } // // Initialize the open device nodes // XID_Globals.DeviceNodes = (PXID_DEVICE_NODE)openDevices; XID_Globals.DeviceNodeInUseCount = 0; for(index = 0; index < XID_Globals.DeviceNodeCount; index++) { XID_Globals.DeviceNodes[index].InUse = FALSE; } // // Register our resources // XID_gResourceRequirements.MaxDevices = handleCount; UsbInit->RegisterResources(&XID_gResourceRequirements); // // Initialize other globals. // KeInitializeTimer(&XID_Globals.EnumWatchdogTimer); return; } #pragma code_seg(".XPPCODE") __inline void XID_SetEnumWatchdog() /*++ Routine Description: Called to start a 5 second watchdog timer on transfers during device enumeration. --*/ { LARGE_INTEGER wait; wait.QuadPart = -50000000; KeSetTimer(&XID_Globals.EnumWatchdogTimer,wait,&XID_Globals.EnumWatchdogTimerDpc); } __inline void XID_ClearEnumWatchdog() /*++ Routine Description: Called to cancel a 5 second watchdog timer when a transfer during enumeration completes. --*/ { KeCancelTimer(&XID_Globals.EnumWatchdogTimer); } VOID XID_WatchdogTimerProc(PKDPC, PVOID pvDevice, PVOID, PVOID) /*++ Routine Description: DPC routine called if a 5 second watchdog timer expires protecting asynchronous transfers during enumeration. The routine forces completion of the URB by canceling it. This will cause enumeration to fail, but avoids hanging USB device enumeration system wide. --*/ { IUsbDevice *device = (IUsbDevice *)pvDevice; USB_DBG_ERROR_PRINT(("Enumeration Watchdog Timer has expired.\n")); device->CancelRequest(&XID_Globals.EnumUrb); } EXTERNUSB VOID XID_AddDevice( IN IUsbDevice *Device ) /*++ Routine Description: This routine is registered as the AddDevice routine for new XID devices. It performs the following tasks: 1) Allocates a XidNode from the static pool of nodes. 2) Begins the XID enumeration provcess. Enumeration is continued in XID_EnumStage1. --*/ { UCHAR nodeIndex = 0; PXID_DEVICE_NODE xidNode; const USB_ENDPOINT_DESCRIPTOR *endpointDescriptor; USB_DBG_ENTRY_PRINT(("Entering XID_AddDevice.\n")); if(XID_Globals.DeviceNodeInUseCount < XID_Globals.DeviceNodeCount) { // // Find a free node, and claim it. // while(XID_Globals.DeviceNodes[nodeIndex].InUse) nodeIndex++; XID_Globals.DeviceNodeInUseCount++; XID_Globals.EnumNode = nodeIndex; xidNode = XID_Globals.DeviceNodes + nodeIndex; Device->SetExtension((PVOID)xidNode); xidNode->Device = Device; // // Initialize what we can about the device. // xidNode->InUse = TRUE; xidNode->Ready = FALSE; xidNode->PendingRemove = FALSE; xidNode->Opened = FALSE; xidNode->InterfaceNumber = Device->GetInterfaceNumber(); xidNode->OpenDevice = NULL; // // Get the interrupt-in endpoint descriptor (NOT OPTIONAL). // endpointDescriptor = Device->GetEndpointDescriptor(USB_ENDPOINT_TYPE_INTERRUPT, TRUE, 0); ASSERT(endpointDescriptor); xidNode->EndpointAddressIn = endpointDescriptor->bEndpointAddress; xidNode->MaxPacketIn = (UCHAR)endpointDescriptor->wMaxPacketSize; // // Get the interrupt-out endpoint (OPTIONAL) // endpointDescriptor = Device->GetEndpointDescriptor(USB_ENDPOINT_TYPE_INTERRUPT, FALSE, 0); if(endpointDescriptor) { xidNode->EndpointAddressOut = endpointDescriptor->bEndpointAddress; xidNode->MaxPacketOut = (UCHAR)endpointDescriptor->wMaxPacketSize; } else { xidNode->EndpointAddressOut = 0; xidNode->MaxPacketOut = 0; } // // Get XID Descriptor // USB_BUILD_CONTROL_TRANSFER( (PURB_CONTROL_TRANSFER)&XID_Globals.EnumUrb, NULL, //Default endpoint (PVOID)&XID_Globals.EnumDeviceDescriptor, sizeof(XID_DESCRIPTOR), USB_TRANSFER_DIRECTION_IN, (PURB_COMPLETE_PROC)XID_EnumStage1, (PVOID)xidNode, TRUE, (USB_DEVICE_TO_HOST | USB_VENDOR_COMMAND | USB_COMMAND_TO_INTERFACE), USB_REQUEST_GET_DESCRIPTOR, XID_DESCRIPTOR_TYPE, xidNode->InterfaceNumber, sizeof(XID_DESCRIPTOR) ); // // Reinitialize the watchdog DPC, so that the context is our device // KeInitializeDpc(&XID_Globals.EnumWatchdogTimerDpc, XID_WatchdogTimerProc, (PVOID)Device); XID_SetEnumWatchdog(); Device->SubmitRequest(&XID_Globals.EnumUrb); } else { // // Out of nodes so we cannot support the device // (allow retry) // ASSERT(FALSE); Device->AddComplete(USBD_STATUS_NO_MEMORY); } USB_DBG_EXIT_PRINT(("Exiting XID_AddDevice.\n")); } VOID XID_EnumStage1( PURB Urb, PXID_DEVICE_NODE XidNode ) /*++ Routine Description: Completion routine for obtaining the XID descriptor started in AddDevice. If retriving the XID_DESCRIPTOR failed, --*/ { XID_ClearEnumWatchdog(); // // Getting the XID Descriptor may have failed. This probably // means it is a legacy device. // if(USBD_ERROR(Urb->Header.Status)) { USB_DBG_EXIT_PRINT(("Couldn't get XID descriptor, could be a HID keyboard.\n")); XID_EnumLegacy1(Urb, XidNode); return; } // // Copy the necessary info out of the Xid Descriptor // XidNode->TypeInformation = GetTypeInformation(XID_Globals.EnumXidDescriptor.bType, &XidNode->TypeIndex); XidNode->SubType = XID_Globals.EnumXidDescriptor.bSubType; XidNode->bMaxInputReportSize = XID_Globals.EnumXidDescriptor.bMaxInputReportSize; XidNode->bMaxOutputReportSize = XID_Globals.EnumXidDescriptor.bMaxOutputReportSize; // // Check a number of items to determine if we can support the device. // 1) The type must be one known at compile time // 2) The report size must be at least as large as the XID_REPORT_HEADER. // 3) The maximum report size must not be larger than our buffer (XID_MAXIMUM_REPORT_SIZE) // this is a limit on the extensibility of XID. // 4) For performance and resource management reasons we enforce that // the entire report fit in a single interrupt-IN packet (there is // no such requirement on the control pipe). // 5) Same for output, but only if interrupt OUT is supported. // if( (!XidNode->TypeInformation) || (XID_REPORT_HEADER > XID_Globals.EnumXidDescriptor.bMaxInputReportSize) || (XID_MAXIMUM_REPORT_SIZE < XID_Globals.EnumXidDescriptor.bMaxInputReportSize) || (XidNode->bMaxInputReportSize > XidNode->MaxPacketIn) || (XidNode->EndpointAddressOut && (XidNode->bMaxOutputReportSize > XidNode->MaxPacketOut)) ) { USB_DBG_WARN_PRINT(("Unsupported XID Device")); XidNode->Device->SetExtension(NULL); XidNode->Device->AddComplete(USBD_STATUS_UNSUPPORTED_DEVICE); XidNode->Device=NULL; XidNode->InUse = FALSE; XID_Globals.DeviceNodeInUseCount--; return; } // // Set the class specific type and call AddComplete. // XidNode->Device->SetClassSpecificType(XidNode->TypeIndex); XidNode->Device->AddComplete(USBD_STATUS_SUCCESS); XidNode->Ready = TRUE; } EXTERNUSB VOID XID_RemoveDevice( IUsbDevice *Device ) /*++ Routine Description: Called by usbd when a device is removed. It kicks off out remove sequence. --*/ { PXID_DEVICE_NODE xidNode = (PXID_DEVICE_NODE)Device->GetExtension(); ASSERT(NULL != xidNode); ASSERT_DISPATCH_LEVEL(); // // Mark the Node remove pending. // xidNode->PendingRemove = TRUE; // // If there is an open handle // kick off a close. // if(xidNode->OpenDevice) { // // Notify keyboard services of device removal. // #ifdef DEBUG_KEYBOARD if((XDEVICE_TYPE_DEBUG_KEYBOARD==xidNode->TypeInformation->XppType) && XID_pKeyboardServices) { XID_pKeyboardServices->pfnRemove(xidNode->OpenDevice); } #endif DEBUG_KEYBOARD XID_fCloseEndpoints(xidNode->OpenDevice); } else // // Otherwise, we can wrap up the remove. // { XID_fRemoveDeviceComplete(xidNode); } } VOID FASTCALL XID_fRemoveDeviceComplete( IN PXID_DEVICE_NODE XidNode ) /*++ Routine Description: This routine is called by XID_DereferenceNode, when the reference count on a XID node goes to zero. At that time, it is certain that all open endpoints are closed, and all outstanding I/O is complete. --*/ { ASSERT_DISPATCH_LEVEL(); // // We are really done // USB_DBG_TRACE_PRINT(("Freeing node(0x%0.8x) nodeIndex = %d\n", XidNode, XidNode - XID_Globals.DeviceNodes)); ASSERT(XidNode->PendingRemove); XidNode->Device->SetExtension(NULL); XidNode->Device->RemoveComplete(); XidNode->Device=NULL; XidNode->InUse = FALSE; XID_Globals.DeviceNodeInUseCount--; return; } PXID_DEVICE_NODE FASTCALL XID_fFindNode( IN PXID_TYPE_INFORMATION TypeInformation, IN ULONG Port ) /*++ Routine Description: Finds a XID node that corresponds to the type and port (port includes port and slot). Will only return nodes that are ready to be open. Return Value: If a suitable node is found, it is returned. Otherwise, NULL is returned. Environement: Assumes DISPATCH_LEVEL. --*/ { UCHAR xidNodeIndex; PXID_DEVICE_NODE xidNode = NULL; // // Brute force search of all the nodes. // for(xidNodeIndex = 0; xidNodeIndex < XID_Globals.DeviceNodeCount; xidNodeIndex++) { if( XID_Globals.DeviceNodes[xidNodeIndex].InUse && (XID_Globals.DeviceNodes[xidNodeIndex].Device->GetPort() == Port) && (XID_Globals.DeviceNodes[xidNodeIndex].TypeInformation == TypeInformation) && XID_Globals.DeviceNodes[xidNodeIndex].Ready && !XID_Globals.DeviceNodes[xidNodeIndex].PendingRemove ){ xidNode = XID_Globals.DeviceNodes + xidNodeIndex; } } return xidNode; } DWORD FASTCALL XID_fOpenDevice( PXID_TYPE_INFORMATION TypeInformation, ULONG Port, PXID_OPEN_DEVICE *OpenDevice, PXINPUT_POLLING_PARAMETERS PollingParameters ) /*++ Given a XidTy creates an OpenDevice, initializes the OpenDevice and opens the endpoints so the device can be used. --*/ { USBD_STATUS status; PXID_OPEN_DEVICE openDevice; PXID_DEVICE_NODE xidNode; KIRQL oldIrql; DWORD errorCode = ERROR_SUCCESS; BOOL partiallyOpen = FALSE; // // By default we want to return NULL. // *OpenDevice = NULL; // // Must be done at high Irql // oldIrql = KeRaiseIrqlToDpcLevel(); // // Find the node // xidNode = XID_fFindNode(TypeInformation, Port); if(!xidNode) { errorCode = ERROR_DEVICE_NOT_CONNECTED; goto exit_open_device; } // // Make sure that it is not already open // if(xidNode->OpenDevice) { errorCode = ERROR_SHARING_VIOLATION; goto exit_open_device; } if(0==TypeInformation->bRemainingHandles) { errorCode = ERROR_OUTOFMEMORY; goto exit_open_device; } // // Allocate memory for the device handle // TypeInformation->bRemainingHandles--; openDevice = XID_AllocateHandle(); // // Initialize the basic stuff // RtlZeroMemory(openDevice, sizeof(XID_OPEN_DEVICE)); openDevice->XidNode = xidNode; openDevice->TypeInformation = TypeInformation; openDevice->AutoPoll = PollingParameters->fAutoPoll ? TRUE : FALSE; // // Tie the xidNode to our handle. // xidNode->OpenDevice = openDevice; // // Any failure after here, requires closing the device. // partiallyOpen = TRUE; // // Open the endpoints // status = XID_fOpenEndpoints(openDevice, PollingParameters); // // If the endpoints were successfully opened // then setup the initial state. // if(USBD_SUCCESS(status)) { // // Initialize the Report and the Report for URB // with the default values; // RtlCopyMemory( (PVOID)(openDevice->ReportForUrb+XID_REPORT_HEADER), TypeInformation->pInputReportInfoList[0].pDefaultValues, TypeInformation->pInputReportInfoList[0].bCurrentSize ); RtlCopyMemory( (PVOID)openDevice->Report, (PVOID)(openDevice->ReportForUrb+XID_REPORT_HEADER), sizeof(openDevice->Report) ); // // Initialize an event for synchronization // KEVENT event; KeInitializeEvent(&event, SynchronizationEvent, FALSE); // // Build a control transfer // WORD wReportTypeAndId = 0x0100; ULONG ulBufferSize = xidNode->bMaxInputReportSize; USB_BUILD_CONTROL_TRANSFER( &openDevice->Urb.ControlTransfer, NULL, //Default endpoint openDevice->ReportForUrb, ulBufferSize, USB_TRANSFER_DIRECTION_IN, (PURB_COMPLETE_PROC)XID_SyncComplete, //do it synchronously &event, //the context is the event. TRUE, (USB_DEVICE_TO_HOST | USB_CLASS_COMMAND | USB_COMMAND_TO_INTERFACE), XID_COMMAND_GET_REPORT, wReportTypeAndId, //Input report 0 xidNode->InterfaceNumber, ulBufferSize ); // // Submit the request // xidNode->Device->SubmitRequest(&openDevice->Urb); // // Lower IRQL, and wait for transfer to complete. // Then restore Irql to DPC level // KeLowerIrql(oldIrql); KeWaitForSingleObject(&event, Executive, KernelMode, FALSE, NULL); oldIrql = KeRaiseIrqlToDpcLevel(); // // During the transfer the device may have been removed, // if so we should check to make sure that the device is // still with us. There are two cases here: the device is // totally gone, the device is still pending remove. // if(NULL == openDevice->XidNode || xidNode->PendingRemove) { errorCode = ERROR_DEVICE_NOT_CONNECTED; goto exit_open_device; } // // Process the GET_REPORT results, if it succeeded // if(USBD_SUCCESS(openDevice->Urb.Header.Status)) { TypeInformation->pfnProcessNewData(openDevice); } #if DBG else // // Issue warning if GET_REPORT failed. // { if(USBD_STATUS_STALL_PID == openDevice->Urb.Header.Status) { USB_DBG_WARN_PRINT(("Device in port %d (0 based) does not support GET_REPORT\n", Port)); } } #endif // // Notify keyboard services of new open device. // if((XDEVICE_TYPE_DEBUG_KEYBOARD==TypeInformation->XppType) && XID_pKeyboardServices) { XID_pKeyboardServices->pfnOpen(openDevice); } // // Prepare the URB for the interrupt-in endpoint. // USB_BUILD_BULK_OR_INTERRUPT_TRANSFER( (PURB_BULK_OR_INTERRUPT_TRANSFER)&openDevice->Urb, openDevice->InterruptInEndpointHandle, openDevice->ReportForUrb, xidNode->bMaxInputReportSize, USB_TRANSFER_DIRECTION_IN, (PURB_COMPLETE_PROC)XID_NewInterruptData, (PVOID)openDevice, TRUE ); // // If autopoll is set, start polling // if(openDevice->AutoPoll) { xidNode->Device->SubmitRequest(&openDevice->Urb); } // // If we are here, the device was successfully opened. // copy the handle, and mark partiallyOpen as FALSE // as we are now fully open. // *OpenDevice = openDevice; partiallyOpen = FALSE; } else { errorCode = IUsbDevice::Win32FromUsbdStatus(status); } exit_open_device: // // Done with synchronization // KeLowerIrql(oldIrql); // // Partially opened devices // should be closed. // if(partiallyOpen) { XID_fCloseDevice(openDevice); } return errorCode; } USBD_STATUS FASTCALL XID_fOpenEndpoints( IN PXID_OPEN_DEVICE OpenDevice, PXINPUT_POLLING_PARAMETERS PollingParameters ) /*++ Routine Description: Called from XID_OpenDevice to open the endpoints associated with the device. Opening the endpoints references the XidNode until the endpoints are closed. --*/ { USBD_STATUS status = USBD_STATUS_SUCCESS; PXID_DEVICE_NODE xidNode = OpenDevice->XidNode; // // Open the control endpoint // USB_BUILD_OPEN_DEFAULT_ENDPOINT((PURB_OPEN_ENDPOINT)&OpenDevice->Urb); status = xidNode->Device->SubmitRequest(&OpenDevice->Urb); // // If the control endpoint is open, // open the interrupt in endpoint. // if(USBD_SUCCESS(status)) { OpenDevice->ControlEndpointOpen = TRUE; USB_BUILD_OPEN_ENDPOINT( (PURB_OPEN_ENDPOINT)&OpenDevice->Urb, xidNode->EndpointAddressIn, USB_ENDPOINT_TYPE_INTERRUPT, (USHORT)XID_MAXIMUM_REPORT_SIZE, PollingParameters->bInputInterval ); status = xidNode->Device->SubmitRequest(&OpenDevice->Urb); } // // If interrupt-IN endpoint opened, save the handle // and try to open the interrupt out, if there is // one. // if(USBD_SUCCESS(status)) { OpenDevice->InterruptInEndpointHandle = OpenDevice->Urb.OpenEndpoint.EndpointHandle; ASSERT(OpenDevice->InterruptInEndpointHandle); if(PollingParameters->fInterruptOut && xidNode->EndpointAddressOut) { USB_BUILD_OPEN_ENDPOINT( (PURB_OPEN_ENDPOINT)&OpenDevice->Urb, xidNode->EndpointAddressOut, USB_ENDPOINT_TYPE_INTERRUPT, (USHORT)XID_MAXIMUM_REPORT_SIZE, PollingParameters->bOutputInterval ); status = xidNode->Device->SubmitRequest(&OpenDevice->Urb); if(USBD_SUCCESS(status)) { OpenDevice->InterruptOutEndpointHandle = OpenDevice->Urb.OpenEndpoint.EndpointHandle; ASSERT(OpenDevice->InterruptOutEndpointHandle); } } } // // return the status // return status; } VOID FASTCALL XID_fCloseDevice( IN PXID_OPEN_DEVICE OpenDevice ) /*++ Routine Description: Closes an open or paritally open device. Always does the right thing. This routine assumes that OpenDevice is valid. --*/ { ASSERT_LESS_THAN_DISPATCH_LEVEL(); KIRQL oldIrql = KeRaiseIrqlToDpcLevel(); // // If we are still attached to // a xidnode, then we need to close the // endpoints. // if(OpenDevice->XidNode) { // // Notify keyboard services of device closing. // #ifdef DEBUG_KEYBOARD if((XDEVICE_TYPE_DEBUG_KEYBOARD==OpenDevice->TypeInformation->XppType) && XID_pKeyboardServices) { XID_pKeyboardServices->pfnClose(OpenDevice); } #endif DEBUG_KEYBOARD // // Setup event to synchronize close // KEVENT event; KeInitializeEvent(&event, NotificationEvent, FALSE); OpenDevice->CloseEvent = &event; // // Mark close pending as true // OpenDevice->ClosePending = TRUE; // // Kick off close endpoint state // machine. // XID_fCloseEndpoints(OpenDevice); // // Wait for close endpoint state machine // to do its job. // KeLowerIrql(oldIrql); KeWaitForSingleObject(&event, Executive, KernelMode, FALSE, NULL); } else { KeLowerIrql(oldIrql); } // // Everything is cleaned up, just free the memory. // OpenDevice->TypeInformation->bRemainingHandles++; XID_FreeHandle(OpenDevice); } VOID FASTCALL XID_fCloseEndpoints( PXID_OPEN_DEVICE OpenDevice ) /*++ Routine Description: This routine closes the endpoints associated with an OpenDevice handle. --*/ { ASSERT_DISPATCH_LEVEL(); // // Guard against rentry // if(OpenDevice->EndpointsPendingClose) { return; } // kill the Alamo DPC timer, if it is running #ifdef ALAMO_RAW_DATA_HACK if(OpenDevice->XidNode->AlamoHack) KeCancelTimer(&OpenDevice->AlamoPollTimer); #endif //ALAMO_RAW_DATA_HACK OpenDevice->EndpointsPendingClose = TRUE; // // Start endpoint state machine. // XID_CloseEndpointStateMachine(&OpenDevice->CloseUrb, OpenDevice); } VOID XID_CloseEndpointStateMachine( PURB_CLOSE_ENDPOINT CloseUrb, PXID_OPEN_DEVICE OpenDevice ) { IUsbDevice *device = OpenDevice->XidNode->Device; // // If the control endpoint is open // build an URB to close it. // if(OpenDevice->ControlEndpointOpen) { // // Close the default endpoint // USB_BUILD_CLOSE_DEFAULT_ENDPOINT( CloseUrb, (PURB_COMPLETE_PROC)XID_CloseEndpointStateMachine, (PVOID)OpenDevice ); OpenDevice->ControlEndpointOpen = FALSE; } else if(OpenDevice->InterruptInEndpointHandle) // // If the interrupt in endpoint needs to be closed // build an URB to close it. // { USB_BUILD_CLOSE_ENDPOINT( CloseUrb, OpenDevice->InterruptInEndpointHandle, (PURB_COMPLETE_PROC)XID_CloseEndpointStateMachine, (PVOID)OpenDevice ); OpenDevice->InterruptInEndpointHandle = NULL; } else if(OpenDevice->InterruptOutEndpointHandle) // // If the interrupt in endpoint needs to be closed // build an URB to close it. // { USB_BUILD_CLOSE_ENDPOINT( CloseUrb, OpenDevice->InterruptOutEndpointHandle, (PURB_COMPLETE_PROC)XID_CloseEndpointStateMachine, (PVOID)OpenDevice ); OpenDevice->InterruptOutEndpointHandle = NULL; } else // // Everything is closed, handle the reason // we close the endpoints (either pending close // or pending remove). // { PXID_DEVICE_NODE xidNode = OpenDevice->XidNode; // // We should only be here if either a close pending // or if a remove is pending. // ASSERT(OpenDevice->ClosePending || xidNode->PendingRemove); // // In either case, we need to sever the xidNode // and the openDevice. // xidNode->OpenDevice = NULL; OpenDevice->XidNode = NULL; // // If there is a pending remove, // we need to call remove complete. // if(xidNode->PendingRemove) { XID_fRemoveDeviceComplete(xidNode); } // // If there is a pending close, // signal the close event. // if(OpenDevice->ClosePending) { // // Signal the close event // ASSERT(OpenDevice->CloseEvent); KeSetEvent(OpenDevice->CloseEvent, IO_NO_INCREMENT, FALSE); } return; } // // submit the next request // device->SubmitRequest((PURB)CloseUrb); } VOID XID_NewInterruptData( PURB Urb, PXID_OPEN_DEVICE OpenDevice ) /*++ Routine Description: This is the completion routine for new input data. We never need to check xidNode or the OpenDevice since this is a completion routine for I/O. I/O is always guaranteed to complete for a close. So we can safely process the I/O. However, before resubmitting any I/O requests, we should verify that a close or remove is not pending. --*/ { ASSERT_DISPATCH_LEVEL(); PXID_DEVICE_NODE xidNode = OpenDevice->XidNode; // // If the device has been closed or is pending // remove, there is nothing to do. // if(OpenDevice->ClosePending || xidNode->PendingRemove) { return; } // // If the URB was completed successfully, then // process the data. // if(USBD_SUCCESS(Urb->Header.Status)) { // // Debug only check that the device is not returning short packets // #if DBG if( Urb->BulkOrInterruptTransfer.TransferBufferLength < xidNode->bMaxInputReportSize) { USB_DBG_TRACE_PRINT(("An input device returned a short packet(expected %d, received %d).", xidNode->bMaxInputReportSize, Urb->BulkOrInterruptTransfer.TransferBufferLength )); } if( Urb->BulkOrInterruptTransfer.TransferBufferLength > xidNode->bMaxInputReportSize) { USB_DBG_ERROR_PRINT(("Packet came back larger than maximum allowed.")); Urb->BulkOrInterruptTransfer.TransferBufferLength = xidNode->bMaxInputReportSize; } #endif // // Process the new data // xidNode->TypeInformation->pfnProcessNewData(OpenDevice); OpenDevice->PacketNumber++; OpenDevice->OutstandingPoll = 0; // // Resubmit the URB if autopoll, reset the transfer // length, in case the last transfer came up short. // Urb->BulkOrInterruptTransfer.TransferBufferLength = xidNode->bMaxInputReportSize; // // If this is not an auto-poll device, we are done // if(!OpenDevice->AutoPoll) { return; } } else { // // Print out the error. // #if DBG if( (USBD_STATUS_CANCELED != Urb->Header.Status) && (USBD_ISOCH_STATUS_DEV_NOT_RESPONDING != Urb->Header.Status) //99% of the time indicates device is gone ) { USB_DBG_WARN_PRINT(("An input device poll failed. Status = 0x%0.8x\n", Urb->Header.Status)); } #endif // // Alter the URB so that it clears the endpoint halt. // USB_BUILD_CLEAR_FEATURE( (PURB_CONTROL_TRANSFER)Urb, USB_COMMAND_TO_ENDPOINT, USB_FEATURE_ENDPOINT_STALL, xidNode->EndpointAddressIn, (PURB_COMPLETE_PROC)XID_ClearInputStallComplete, OpenDevice ); } // // Resubmit the URB // xidNode->Device->SubmitRequest(Urb); } VOID XID_ClearInputStallComplete( PURB Urb, PXID_OPEN_DEVICE OpenDevice ) /*++ Routine Description: This is the completion routine for sendinf a CLEAR_FEATURE(ENDPOINT_HALT). We never need to check xidNode or the OpenDevice since this is a completion routine for I/O. I/O is always guaranteed to complete for a close. We should check to see if a close or remove is pending before trying to finish the endpoint reset, or submitting an input URB. --*/ { PXID_DEVICE_NODE xidNode = OpenDevice->XidNode; USBD_STATUS status; // // Don't do anything if a remove or close is pending. // if(xidNode->PendingRemove || OpenDevice->ClosePending) { return; } // // If the CLEAR_FEATURE succeeded, // reset the endpoint state, and try to resubmit // the I/O. // if(USBD_SUCCESS(Urb->Header.Status)) { // // Build URB to reset the endpoint state // USB_BUILD_SET_ENDPOINT_STATE( (PURB_GET_SET_ENDPOINT_STATE)Urb, OpenDevice->InterruptInEndpointHandle, USB_ENDPOINT_STATE_CLEAR_HALT | USB_ENDPOINT_STATE_DATA_TOGGLE_RESET ); status = xidNode->Device->SubmitRequest(Urb); ASSERT(USBD_SUCCESS(status)); USB_DBG_WARN_PRINT(("Reset endpoint, resume polling.\n")); // // Build URB to resume polling // #ifdef ALAMO_RAW_DATA_HACK if(xidNode->AlamoHack) XID_AlamoStartPoll(OpenDevice); else { #endif USB_BUILD_BULK_OR_INTERRUPT_TRANSFER( (PURB_BULK_OR_INTERRUPT_TRANSFER)&OpenDevice->Urb, OpenDevice->InterruptInEndpointHandle, OpenDevice->ReportForUrb, xidNode->bMaxInputReportSize, USB_TRANSFER_DIRECTION_IN, (PURB_COMPLETE_PROC)XID_NewInterruptData, (PVOID)OpenDevice, TRUE ); xidNode->Device->SubmitRequest(Urb); #ifdef ALAMO_RAW_DATA_HACK } #endif } else { USB_DBG_WARN_PRINT(("Failed to clear halt. Report device as not responding. Status = 0x%0.8x\n", Urb->Header.Status)); xidNode->Device->DeviceNotResponding(); } } void FASTCALL XID_ProcessGamepadData( PXID_OPEN_DEVICE OpenDevice ) /*++ Routine Description: Processing of gamepad data. Basically, it copies the data from the buffer that is ping-ponged to the hardware, to the one that is always available for copying from XInputGetState. Then it checks to see if anything changes and hits XAutoPowerDownResetTimer --*/ { XINPUT_GAMEPAD *pGamePad = (XINPUT_GAMEPAD *)OpenDevice->Report; if(OpenDevice->Urb.CommonTransfer.TransferBufferLength >= XID_REPORT_HEADER) { RtlCopyMemory( (PVOID)pGamePad, (PVOID)(OpenDevice->ReportForUrb+XID_REPORT_HEADER), OpenDevice->Urb.ControlTransfer.TransferBufferLength-XID_REPORT_HEADER ); // // XAutoPowerDownResetTimer if necessary, too much jitter the thumbsticks // means we check all the other controls to see if any of them are pressed. // if( pGamePad->wButtons& (XINPUT_GAMEPAD_DPAD_UP|XINPUT_GAMEPAD_DPAD_DOWN|XINPUT_GAMEPAD_DPAD_LEFT| XINPUT_GAMEPAD_DPAD_RIGHT|XINPUT_GAMEPAD_START|XINPUT_GAMEPAD_BACK) ) { XAutoPowerDownResetTimer(); } else { for(int i = XINPUT_GAMEPAD_A; i <= XINPUT_GAMEPAD_RIGHT_TRIGGER; i++) { if(pGamePad->bAnalogButtons[i]) { XAutoPowerDownResetTimer(); break; } } } } } void FASTCALL XID_ProcessIRRemoteData( PXID_OPEN_DEVICE OpenDevice ) /*++ Routine Description: IRREM XID processing of new data. Copies the data from the buffer that is ping-ponged to the hardware, then it hits XAutoPowerDownResetTimer. --*/ { RtlCopyMemory( (PVOID)OpenDevice->Report, (PVOID)(OpenDevice->ReportForUrb+XID_REPORT_HEADER), OpenDevice->Urb.ControlTransfer.TransferBufferLength-XID_REPORT_HEADER ); XAutoPowerDownResetTimer(); } void FASTCALL XID_ProcessNewKeyboardData( PXID_OPEN_DEVICE OpenDevice ) /*++ Routine Description: Keyboard processing is slightly different. It doesn't have a XID_REPORT_HEADER --*/ { RtlCopyMemory(OpenDevice->Report, OpenDevice->ReportForUrb, sizeof(XINPUT_KEYBOARD)); if(XID_pKeyboardServices) { XID_pKeyboardServices->pfnNewData((PVOID)OpenDevice, (PXINPUT_KEYBOARD)OpenDevice->Report); XAutoPowerDownResetTimer(); } } DWORD FASTCALL XID_fSendDeviceReport( IN PXID_OPEN_DEVICE OpenDevice, IN PXINPUT_FEEDBACK_INTERNAL OutputReport ) { PVOID rawOutputBuffer; USBD_STATUS usbStatus; PXID_DEVICE_NODE xidNode = OpenDevice->XidNode; KIRQL oldIrql = KeRaiseIrqlToDpcLevel(); // // Check to make sure the device has not been removed // if(NULL==xidNode || xidNode->PendingRemove) { OutputReport->dwStatus = ERROR_DEVICE_NOT_CONNECTED; goto exit_send_device_report; } // // If bMaxOutputReportSize is zero, the device doesn't support output. // if(0 == xidNode->bMaxOutputReportSize) { OutputReport->dwStatus = ERROR_NOT_SUPPORTED; goto exit_send_device_report; } // // Increase the reference count on the completion event // if there is one. // if(OutputReport->hEvent) { NTSTATUS status = ObReferenceObjectByHandle( OutputReport->hEvent, ExEventObjectType, (PHANDLE)&OutputReport->Internal.CompletionEvent ); // // If the following assertion fails the event handle is invalid. // This does not prevent the I/O, but we cannot set the event. // We assert, and NULL out both the hEvent and the pointer // we would have obtained from ObReferenceObjectByHandle // //ASSERT(NT_SUCCESS(status)); if(!NT_SUCCESS(status)) { USB_DBG_ERROR_PRINT(("ERROR: pFeedback->Header.hEvent (0x%0.8x) is invalid.\n" "hEvent should be NULL or a valid Event handle.\n" "Event cannot be signaled at completion.\n", OutputReport->hEvent)); OutputReport->hEvent = NULL; OutputReport->Internal.CompletionEvent = NULL; } } else { OutputReport->Internal.CompletionEvent = NULL; } // // Constrain the size if needed. // if(xidNode->bMaxOutputReportSize < OutputReport->Internal.bSize) { OutputReport->Internal.bSize = xidNode->bMaxOutputReportSize; } // // Legacy devices (notably keyboard) does not have a XID_REPORT_HEADER. // if(xidNode->TypeInformation->ulFlags&XID_BSF_NO_OUTPUT_HEADER) { rawOutputBuffer = (PVOID)(&OutputReport->Internal.bReportId + XID_REPORT_HEADER); } else // // Otherwise rawBuffer starts at the bReportId of the internal header // { rawOutputBuffer = (PVOID)&OutputReport->Internal.bReportId; } // // Build the URB (use the interrupt Out pipe if there is // one, otherwise use the default pipe with SET_REPORT). // if(OpenDevice->InterruptOutEndpointHandle) { USB_BUILD_BULK_OR_INTERRUPT_TRANSFER( &OutputReport->Internal.Urb.BulkOrInterruptTransfer, OpenDevice->InterruptOutEndpointHandle, rawOutputBuffer, OutputReport->Internal.bSize, USB_TRANSFER_DIRECTION_OUT, (PURB_COMPLETE_PROC)XID_OutputComplete, (PVOID)OutputReport, FALSE ); } else { USB_BUILD_CONTROL_TRANSFER( &OutputReport->Internal.Urb.ControlTransfer, NULL, rawOutputBuffer, OutputReport->Internal.bSize, USB_TRANSFER_DIRECTION_OUT, (PURB_COMPLETE_PROC)XID_OutputComplete, (PVOID)OutputReport, FALSE, (USB_HOST_TO_DEVICE | USB_CLASS_COMMAND | USB_COMMAND_TO_INTERFACE), XID_COMMAND_SET_REPORT, 0x0200 | OutputReport->Internal.bReportId, xidNode->InterfaceNumber, (WORD)OutputReport->Internal.bSize ); } // // Submit the request // OutputReport->Internal.OpenDevice = OpenDevice; usbStatus = xidNode->Device->SubmitRequest(&OutputReport->Internal.Urb); OutputReport->dwStatus = IUsbDevice::Win32FromUsbdStatus(usbStatus); exit_send_device_report: KeLowerIrql(oldIrql); return OutputReport->dwStatus; } VOID XID_OutputComplete( PURB Urb, PXINPUT_FEEDBACK_INTERNAL OutputReport ) { ASSERT_DISPATCH_LEVEL(); // // Check to see if the device has been removed // PXID_OPEN_DEVICE openDevice = OutputReport->Internal.OpenDevice; PXID_DEVICE_NODE xidNode = openDevice->XidNode; // // If the device has been removed or closed // set the error to USBD_STATUS_NO_DEVICE // if(openDevice->ClosePending || xidNode->PendingRemove) { Urb->Header.Status = USBD_STATUS_NO_DEVICE; } // // If the interrupt-Out endpoint is stalled we have to clear the stall, // prior to copying the status and signalling an event. // if( (Urb->Header.Status == USBD_STATUS_STALL_PID) && (Urb->Header.Function == URB_FUNCTION_BULK_OR_INTERRUPT_TRANSFER) ) { // // Print out the error. // USB_DBG_WARN_PRINT(("The interrupt-Out endpoint stalled, reseting.\n", Urb->Header.Status)); // // Alter the URB so that it clears the endpoint halt. // USB_BUILD_CLEAR_FEATURE( (PURB_CONTROL_TRANSFER)Urb, USB_COMMAND_TO_ENDPOINT, USB_FEATURE_ENDPOINT_STALL, xidNode->EndpointAddressOut, (PURB_COMPLETE_PROC)XID_ClearOutputStallComplete, OutputReport ); // // Submit the request to clean the stall. // xidNode->Device->SubmitRequest(Urb); } else // // If the endpoint is not stalled (or it was a control endpoint which does not need clearing) // go ahead and complete the output. // { XID_fOutputComplete1(Urb, OutputReport); } } VOID FASTCALL XID_fOutputComplete1( PURB Urb, PXINPUT_FEEDBACK_INTERNAL OutputReport ) { PKEVENT event = OutputReport->Internal.CompletionEvent; // // Copy and convert the status. // OutputReport->dwStatus = IUsbDevice::Win32FromUsbdStatus(Urb->Header.Status); // // If the call provided a completion event, set it // and then dereference it. // if(event) { KeSetEvent(event, IO_NO_INCREMENT, FALSE); ObDereferenceObject(event); } } VOID XID_ClearOutputStallComplete( PURB Urb, PXINPUT_FEEDBACK_INTERNAL OutputReport ) /*++ Routine Description: This is the completion routine for sendinf a CLEAR_FEATURE(ENDPOINT_HALT). We never need to check xidNode or the OpenDevice since this is a completion routine for I/O. I/O is always guaranteed to complete for a close. We should check to see if a close or remove is pending before trying to finish the endpoint reset, or submitting an input URB. --*/ { PXID_OPEN_DEVICE openDevice = OutputReport->Internal.OpenDevice; PXID_DEVICE_NODE xidNode = openDevice->XidNode; USBD_STATUS status = Urb->Header.Status; // // If the device has been removed or closed // set the error to USBD_STATUS_NO_DEVICE // if(openDevice->ClosePending || xidNode->PendingRemove) { Urb->Header.Status = USBD_STATUS_NO_DEVICE; } else { if(USBD_SUCCESS(status)) { // // Build and submit URB to reset the endpoint state // USB_BUILD_SET_ENDPOINT_STATE( (PURB_GET_SET_ENDPOINT_STATE)Urb, openDevice->InterruptOutEndpointHandle, USB_ENDPOINT_STATE_CLEAR_HALT | USB_ENDPOINT_STATE_DATA_TOGGLE_RESET ); xidNode->Device->SubmitRequest(Urb); } else { USB_DBG_ERROR_PRINT(("Attempt to clear endpoint stall result in status: 0x%0.8x.\n", status)); } Urb->Header.Status = USBD_STATUS_STALL_PID; } XID_fOutputComplete1(Urb, OutputReport); } VOID XID_SyncComplete( PURB /*UnreferencedUrb*/, PKEVENT SyncEvent ) // // Used to make an asyncronous call, synchronous. // { KeSetEvent(SyncEvent, IO_NO_INCREMENT, FALSE); } //**** //**** Legacy Hardware Enumeration (right now the //**** Microsoft Natrual Keyboard Pro is the only supported //**** legacy device). Use of this device is limited for //**** use as a development aid. However, this is a policy //**** decision, not a technical one. Basically, we //**** will always enumerate a Natural Keyboard Pro, //**** however several required methods for using it //**** require lining xdbg.lib, which is not allowed for //**** shipping titles. //**** #define VID_MICROSOFT 0x045E #define PID_NATURAL_KEYBOARD_PRO 0x001D VOID XID_EnumLegacy1( PURB Urb, PXID_DEVICE_NODE XidNode ) /*++ Routine Description: Gets the full device descriptor. The gives us access to the VID and PID so we can support a particular device. Comment: The commented out code below allows the support of any known protocol device (i.e. any boot keyboard). This was scratched and replaced with VID\PID detection of the Microsoft Natural Keyboard Pro just for support reasons. The other code should work on a wide variety of keyboards. --*/ { USB_DBG_ENTRY_PRINT(("Entering XID_EnumLegacy1.\n")); // // First see if it is a known protocol device. (a.k.a. keyboard or // mouse, but we don't support mouse at this point) // const USB_INTERFACE_DESCRIPTOR *interfaceDescriptor; interfaceDescriptor = XidNode->Device->GetInterfaceDescriptor(); if( (USB_DEVICE_CLASS_HUMAN_INTERFACE == interfaceDescriptor->bInterfaceClass) && (HID_KEYBOARD_PROTOCOL == interfaceDescriptor->bInterfaceProtocol) ) { // // Switch to the boot protcol // USB_BUILD_CONTROL_TRANSFER( &XID_Globals.EnumUrb.ControlTransfer, NULL, NULL, 0, 0, (PURB_COMPLETE_PROC)XID_EnumKeyboard, (PVOID)XidNode, TRUE, USB_HOST_TO_DEVICE | USB_CLASS_COMMAND | USB_COMMAND_TO_INTERFACE, HID_SET_PROTOCOL, HID_BOOT_PROTOCOL, XidNode->InterfaceNumber, 0); } #if NEVER // // Check for legacy devices that we support via the VID\PID\FW, // currently there are none, so this if #if NEVER. else { // // Get the full device descriptor. // The core driver never bothers to get // more than the first 8 bytes. // USB_BUILD_GET_DESCRIPTOR( &XID_Globals.EnumUrb.ControlTransfer, USB_DEVICE_DESCRIPTOR_TYPE, 0, 0, (PVOID)&XID_Globals.EnumDeviceDescriptor, sizeof(USB_DEVICE_DESCRIPTOR), (PURB_COMPLETE_PROC)XID_EnumLegacy2, (PVOID)XidNode ); } #else else { // // The device is not supported, free the node. // XidNode->Device->SetExtension(NULL); XidNode->Device->AddComplete(USBD_STATUS_UNSUPPORTED_DEVICE); XidNode->Device=NULL; XidNode->InUse = FALSE; XID_Globals.DeviceNodeInUseCount--; return; } #endif XID_SetEnumWatchdog(); XidNode->Device->SubmitRequest(&XID_Globals.EnumUrb); } #if NEVER // // This routine process enumeration of devices based on VID\PID, currently // we have none, so the routine is #if NEVER // VOID XID_EnumLegacy2( PURB Urb, PXID_DEVICE_NODE XidNode ) /*++ Routine Description: Completion routine for XID_EnumLegacy1. At this point we can check the VID PID and decide whether or not to support the device. --*/ { USB_DBG_ENTRY_PRINT(("Entering XID_EnumLegacy2.\n")); XID_ClearEnumWatchdog(); // // Check for legacy devices that we support // // // Natural Keyboard Pro Support (NOT NEEDED, it is a standard boot keyboard) // left as an example of what to do. // if( (VID_MICROSOFT == XID_Globals.EnumDeviceDescriptor.idVendor) && (PID_NATURAL_KEYBOARD_PRO == XID_Globals.EnumDeviceDescriptor.idProduct) ) { const USB_INTERFACE_DESCRIPTOR *interfaceDescriptor; interfaceDescriptor = XidNode->Device->GetInterfaceDescriptor(); // // Only the keyboard interface of this is device is supported // (it has more than one interface) if( (USB_DEVICE_CLASS_HUMAN_INTERFACE == interfaceDescriptor->bInterfaceClass) && (HID_KEYBOARD_PROTOCOL == interfaceDescriptor->bInterfaceProtocol) ) { // // Switch to the boot protcol // USB_BUILD_CONTROL_TRANSFER( &Urb->ControlTransfer, NULL, NULL, 0, 0, (PURB_COMPLETE_PROC)XID_EnumKeyboard, (PVOID)XidNode, TRUE, USB_HOST_TO_DEVICE | USB_CLASS_COMMAND | USB_COMMAND_TO_INTERFACE, HID_SET_PROTOCOL, HID_BOOT_PROTOCOL, XidNode->InterfaceNumber, 0); XID_SetEnumWatchdog(); XidNode->Device->SubmitRequest(&XID_Globals.EnumUrb); // // Keyboard has more enum steps // return; } } // // IF the device is not supported, free the node. // XidNode->Device->SetExtension(NULL); XidNode->Device->AddComplete(USBD_STATUS_UNSUPPORTED_DEVICE); XidNode->Device=NULL; XidNode->InUse = FALSE; XID_Globals.DeviceNodeInUseCount--; } #endif VOID XID_EnumKeyboard( PURB Urb, PXID_DEVICE_NODE XidNode ) /*++ Routine Description: After a keyboard is found there are still more steps. In particular, we must switch to the boot protocol. XID_EnumLegacy2 did that, and this is the completion routine. Now we must set the idle rate to infinite. Infinite should be the default for boot protocol keyboards, but this is for good measure. --*/ { XID_ClearEnumWatchdog(); // // Switch the idle rate to infinite // USB_BUILD_CONTROL_TRANSFER( &XID_Globals.EnumUrb.ControlTransfer, NULL, NULL, 0, 0, (PURB_COMPLETE_PROC)XID_EnumKeyboardComplete, (PVOID)XidNode, TRUE, USB_HOST_TO_DEVICE | USB_CLASS_COMMAND | USB_COMMAND_TO_INTERFACE, HID_SET_IDLE, HID_IDLE_INFINITE, XidNode->InterfaceNumber, 0); XID_SetEnumWatchdog(); XidNode->Device->SubmitRequest(&XID_Globals.EnumUrb); } VOID XID_EnumKeyboardComplete( PURB Urb, PXID_DEVICE_NODE XidNode ) /*++ Routine Description: We are done with enumeration steps for enumerating a keyboard --*/ { USB_DBG_ENTRY_PRINT(("Entering XID_EnumKeyboardComplete.\n")); XID_ClearEnumWatchdog(); // Since keyboards are legacy, we hardcode the type. XidNode->TypeInformation = GetTypeInformation(XID_DEVTYPE_KEYBOARD, &XidNode->TypeIndex); if(!XidNode->TypeInformation) { USB_DBG_WARN_PRINT(("No keyboard support in this build.")); XidNode->Device->SetExtension(NULL); XidNode->Device->AddComplete(USBD_STATUS_UNSUPPORTED_DEVICE); XidNode->Device=NULL; XidNode->InUse = FALSE; XID_Globals.DeviceNodeInUseCount--; return; } XidNode->SubType = XINPUT_DEVSUBTYPE_KBD_STANDARD; XidNode->bMaxInputReportSize = sizeof(XINPUT_KEYBOARD); XidNode->bMaxOutputReportSize = sizeof(XINPUT_KEYBOARD_LEDS); XidNode->Device->SetClassSpecificType(XidNode->TypeIndex); XidNode->Device->AddComplete(USBD_STATUS_SUCCESS); XidNode->Ready = TRUE; }