/*++ Copyright (c) 1996-2000 Microsoft Corporation Module Name: hawk.CPP Abstract: Environment: kernel mode Revision History: --*/ //***************************************************************************** // I N C L U D E S //***************************************************************************** #include "hawk2.h" #include "xvocver.h" #if DBG || DEBUG_LOG ULONG HawkDebugLevel=2; // Level of debug output #endif //------------------------------------------------------------------------------ // Declaration of static tables //------------------------------------------------------------------------------ XHawkMediaObject *XHawkMediaObject::sm_pMicrophones = NULL; XHawkMediaObject *XHawkMediaObject::sm_pHeadphones = NULL; USHORT XHawkMediaObject::sm_AvailableHeadphoneResources = 0; USHORT XHawkMediaObject::sm_AvailableMicrophoneResources = 0; PHAWK_STREAMING_RESOURCES XHawkMediaObject::sm_pFreeStreamingResources = NULL; HAWK_SAMPLE_RATE_INFORMATION XHawkMediaObject::sm_HawkSampleRates[HAWK_SAMPLE_RATE_COUNT] = { { 8000, 16, 0}, // 8.000 kHz {11025, 22, 40}, //11.025 kHz {16000, 32, 0}, //16.000 kHz {22050, 44, 20}, //22.050 kHz {24000, 48, 0} //24.000 kHz }; PUCHAR XHawkMediaObject::sm_pRateIndices = NULL; PUCHAR XHawkMediaObject::sm_pRateIndexRefCounts = NULL; //------------------------------------------------------------------------------ // Declare Hawk types and class. //------------------------------------------------------------------------------ DECLARE_XPP_TYPE(XDEVICE_TYPE_VOICE_MICROPHONE) DECLARE_XPP_TYPE(XDEVICE_TYPE_VOICE_HEADPHONE) USB_DEVICE_TYPE_TABLE_BEGIN(Hawk) USB_DEVICE_TYPE_TABLE_ENTRY(XDEVICE_TYPE_VOICE_MICROPHONE), USB_DEVICE_TYPE_TABLE_ENTRY(XDEVICE_TYPE_VOICE_HEADPHONE) USB_DEVICE_TYPE_TABLE_END() USB_CLASS_DRIVER_DECLARATION(Hawk, USB_DEVICE_CLASS_AUDIO, 0xFF, 0xFF) USB_CLASS_DRIVER_DECLARATION_DUPLICATE(Hawk, 1, XBOX_DEVICE_CLASS_AUDIO_DEVICE, 0xFF, 0xFF) #pragma data_seg(".XPP$ClassHawk") USB_CLASS_DECLARATION_POINTER(Hawk) USB_CLASS_DECLARATION_POINTER_DUPLICATE(Hawk, 1) #pragma data_seg(".XPP$Data") USB_RESOURCE_REQUIREMENTS HAWK_gResourceRequirements = {USB_CONNECTOR_TYPE_HIGH_POWER, 0, 2, 1, 0, 0, 0, 0, 2, HAWK_ISOCH_MAX_ATTACH_BUFFERS}; //------------------------------------------------------------------------------ // USB Class Driver Interface //------------------------------------------------------------------------------ #pragma code_seg(".XPPCINIT") EXTERNUSB VOID HawkInit (IUsbInit *pUsbInit) { DBGPRINT(3, ("enter: HawkInit\n")); ULONG MicrophoneCount; ULONG HeadphoneCount; //sentry against double init, required because we have two driver //table entries (look in the tables above) - this is still in there to support some older firmware. #pragma BUGBUG("Supporting USB_DEVICE_CLASS_AUDIO for old Hawk FW, we don't support any USB audio, so this must go at some point.") static BOOL fInitialized = FALSE; if(fInitialized) return; fInitialized = TRUE; if(pUsbInit->UseDefaultCount()) { MicrophoneCount = HAWK_DEFAULT_MAX_OPENED; HeadphoneCount = HAWK_DEFAULT_MAX_OPENED; } else { MicrophoneCount = pUsbInit->GetMaxDeviceTypeCount(XDEVICE_TYPE_VOICE_MICROPHONE); HeadphoneCount = pUsbInit->GetMaxDeviceTypeCount(XDEVICE_TYPE_VOICE_HEADPHONE); ULONG ulMaxPorts = XGetPortCount(); RIP_ON_NOT_TRUE_WITH_MESSAGE(MicrophoneCount <= ulMaxPorts, "XInitDevices: requested more XDEVICE_TYPE_VOICE_MICROPHONE than available ports."); if(MicrophoneCount > ulMaxPorts) MicrophoneCount = ulMaxPorts; RIP_ON_NOT_TRUE_WITH_MESSAGE(HeadphoneCount <= ulMaxPorts, "XInitDevices: requested more XDEVICE_TYPE_VOICE_HEADPHONE than available ports."); if(HeadphoneCount > ulMaxPorts) HeadphoneCount = ulMaxPorts; } // // Register for microphones and headphone // HAWK_gResourceRequirements.MaxDevices = (UCHAR)MicrophoneCount; pUsbInit->RegisterResources(&HAWK_gResourceRequirements); HAWK_gResourceRequirements.MaxDevices = (UCHAR)HeadphoneCount; pUsbInit->RegisterResources(&HAWK_gResourceRequirements); XHawkMediaObject::InitializeClass(MicrophoneCount, HeadphoneCount); DBGPRINT(3, ("exit: HawkInit\n")); return; } #pragma code_seg(".XPPCODE") EXTERNUSB VOID HawkAddDevice( IN IUsbDevice *Device ) { const USB_ENDPOINT_DESCRIPTOR *pEndpointDescriptor; XHawkMediaObject *pHawkMediaObject = NULL; DWORD dwPort; // // Get the device port and slot // dwPort = Device->GetPort(); if(dwPort >= 16) { DBGPRINT(2,("AddDevice: dwPort %d, bottom slot, not supported!\n", dwPort)); Device->AddComplete(USBD_STATUS_UNSUPPORTED_DEVICE); return; } // // get an endpoint descriptor for the isoch endpoint // (we don't know the direction (microphone or // headphone). Try input first. // pEndpointDescriptor = Device->GetEndpointDescriptor( USB_ENDPOINT_TYPE_ISOCHRONOUS, TRUE, 0); if(pEndpointDescriptor) { pHawkMediaObject = &XHawkMediaObject::sm_pMicrophones[dwPort]; pHawkMediaObject->m_fMicrophone = TRUE; } else { pEndpointDescriptor = Device->GetEndpointDescriptor( USB_ENDPOINT_TYPE_ISOCHRONOUS, FALSE, 0); if(pEndpointDescriptor) { pHawkMediaObject = &XHawkMediaObject::sm_pHeadphones[dwPort]; pHawkMediaObject->m_fMicrophone = FALSE; } else { ASSERT(FALSE && "Isoch endpoint not found"); Device->AddComplete(USBD_STATUS_UNSUPPORTED_DEVICE); return; } } // // Check and initialize the state // pHawkMediaObject->m_fConnected = TRUE; // // Use XHawkMediaObject as the device extension // Device->SetExtension(pHawkMediaObject); pHawkMediaObject->m_pDevice = Device; // // Record the endpoint address, and wMaxPacket // pHawkMediaObject->m_bEndpointAddress = pEndpointDescriptor->bEndpointAddress; pHawkMediaObject->m_wMaxPacket = pEndpointDescriptor->wMaxPacketSize; DBGPRINT(3,("IsochMaxPacket = %x, bEndpointAddress = 0x%02x\n, Interface = %d\n", (ULONG)pEndpointDescriptor->wMaxPacketSize, (ULONG)pEndpointDescriptor->bEndpointAddress, Device->GetInterfaceNumber() )); // // Set the class specific type so it can be enumerated // Device->SetClassSpecificType( pHawkMediaObject->m_fMicrophone ? HAWK_MICROPHONE_CLASS_SPECIFIC_TYPE : HAWK_HEADPHONE_CLASS_SPECIFIC_TYPE ); Device->AddComplete(USBD_STATUS_SUCCESS); } EXTERNUSB VOID HawkRemoveDevice ( IN IUsbDevice *Device ) { XHawkMediaObject *pHawkMediaObject = (XHawkMediaObject *)Device->GetExtension(); // // Mark the device as not connected // pHawkMediaObject->m_fConnected = FALSE; // // If the device is opened, we cannot call // remove complete, because there is // an endpoint open (or in the process // of being closed.) // // // Mark the media object remove pending, // pHawkMediaObject->m_fRemovePending = TRUE; // // If the device is ready, we should close the endpoints // if(pHawkMediaObject->m_fReady) { // // If the endpoint is not in the process // of being closed, then start that process // pHawkMediaObject->CloseEndpoint(); } else // // The media object wasn't opened, so we can // call remove complete // { // // If a close is pending then we // cannot call RemoveComplete, but // when the close completes it will // see the m_fRemovePending flag. // if(!pHawkMediaObject->m_fClosePending) { pHawkMediaObject->m_pDevice = NULL; Device->SetExtension(NULL); Device->RemoveComplete(); pHawkMediaObject->m_fRemovePending = FALSE; } } } //------------------------------------------------------------------------------ // We do not include the CRT, but we need new. // we declare it static, so we don't conflict on linking // with other modules overriding new. //------------------------------------------------------------------------------ #pragma warning(disable:4211) static __inline void * __cdecl operator new(size_t memSize) { return ExAllocatePoolWithTag( memSize, 'kwah' ); } static __inline void __cdecl operator delete(void *pMemory) { ExFreePool(pMemory); } #pragma warning(default:4211) //------------------------------------------------------------------------------ // XHawkMediaObject //------------------------------------------------------------------------------ #pragma code_seg(".XPPCINIT") void XHawkMediaObject::InitializeClass(ULONG ulMicrophoneCount, ULONG ulHeadphoneCount) { // // Allocate a microphone and a headphone // instance for each of the ports up front. // ULONG ulPortCount = XGetPortCount(); ULONG ulTotalStreamingResources = ulHeadphoneCount + ulMicrophoneCount; PHAWK_STREAMING_RESOURCES pStreamingResourceArray; // // Handle insane case, where the user requested no microphone or headphones, but // some how we are here anyway. // if(ulTotalStreamingResources) pStreamingResourceArray = new HAWK_STREAMING_RESOURCES[ulTotalStreamingResources]; sm_pHeadphones = new XHawkMediaObject[ulPortCount]; sm_pMicrophones = new XHawkMediaObject[ulPortCount]; sm_pRateIndices = new UCHAR[ulPortCount]; sm_pRateIndexRefCounts = new UCHAR[ulPortCount]; RIP_ON_NOT_TRUE_WITH_MESSAGE(sm_pHeadphones, "XInitDevices failed due to insufficent memory. There is no graceful recovery."); RIP_ON_NOT_TRUE_WITH_MESSAGE(sm_pMicrophones, "XInitDevices failed due to insufficent memory. There is no graceful recovery."); RIP_ON_NOT_TRUE_WITH_MESSAGE(sm_pRateIndices, "XInitDevices failed due to insufficent memory. There is no graceful recovery."); RIP_ON_NOT_TRUE_WITH_MESSAGE(sm_pRateIndexRefCounts, "XInitDevices failed due to insufficent memory. There is no graceful recovery."); RIP_ON_NOT_TRUE_WITH_MESSAGE(pStreamingResourceArray, "XInitDevices failed due to insufficent memory. There is no graceful recovery."); // // Make a free list out of pStreamingResourceArray // sm_pFreeStreamingResources = NULL; for(ULONG i=0; i < ulTotalStreamingResources; i++) { pStreamingResourceArray[i].pNextFree = sm_pFreeStreamingResources; sm_pFreeStreamingResources = &pStreamingResourceArray[i]; } // // Initialize the number of available microphone and headphone resources // sm_AvailableMicrophoneResources = (USHORT)ulMicrophoneCount; sm_AvailableHeadphoneResources = (USHORT)ulHeadphoneCount; // // Initially none of the rates are set. // RtlZeroMemory(sm_pRateIndexRefCounts, sizeof(UCHAR)*ulPortCount); // // We could loop over them and morph them into // microphones and headphones, but we might // as well leave that until an AddDevice. // } #pragma code_seg(".XPPCODE") STDMETHODIMP_(ULONG) XHawkMediaObject::AddRef (void) { return (ULONG) InterlockedIncrement(&m_lRefCount); } STDMETHODIMP_(ULONG) XHawkMediaObject::Release (void) { LONG lRefCount; lRefCount = InterlockedDecrement(&m_lRefCount); // // Check for decrement below zero // if(0>lRefCount) { DBGPRINT(1, ("XHawkMediaObject: Reference Count Error: Count went below 0!\n")); return 0; } else if (0==lRefCount) { FreeStreamingResources(); } return (ULONG)lRefCount; } STDMETHODIMP XHawkMediaObject::GetInfo ( OUT PXMEDIAINFO pInfo ) { // // If this assert fails the reference count is hosed // or the caller called us after releasing the last // reference. // ASSERT(m_pStreamingResources); // // align values to our block size. In gets CEILed, out gets FLOORed // if (m_fMicrophone) { pInfo->dwFlags = XMO_STREAMF_FIXED_SAMPLE_SIZE | XMO_STREAMF_OUTPUT_ASYNC; pInfo->dwOutputSize = m_pStreamingResources->BytesPerUSBFrame * HAWK_MIN_PACKET_FRAMES; pInfo->dwInputSize = 0; } else { pInfo->dwFlags = XMO_STREAMF_FIXED_SAMPLE_SIZE | XMO_STREAMF_INPUT_ASYNC; pInfo->dwInputSize = m_pStreamingResources->BytesPerUSBFrame * HAWK_MIN_PACKET_FRAMES; pInfo->dwOutputSize = 0; } // // Look ahead is 0. // pInfo->dwMaxLookahead = 0; return S_OK; } STDMETHODIMP XHawkMediaObject::Flush (void) { return S_OK; } STDMETHODIMP XHawkMediaObject::Discontinuity (void) { return S_OK; } STDMETHODIMP XHawkMediaObject::GetStatus ( OUT DWORD *pdwFlags ) { RIP_ON_NOT_TRUE("XHawkMediaObject::GetStatus", NULL!=pdwFlags); KIRQL oldIrql = KeRaiseIrqlToDpcLevel(); ASSERT(m_pStreamingResources); if( m_fReady && m_pStreamingResources->Free.GetHead()) { if(m_fMicrophone) { *pdwFlags = XMO_STATUSF_ACCEPT_OUTPUT_DATA; } else { *pdwFlags = XMO_STATUSF_ACCEPT_INPUT_DATA; } } else { *pdwFlags = 0; } KeLowerIrql(oldIrql); return S_OK; } HRESULT XHawkMediaObject::Process ( IN LPCXMEDIAPACKET pInputPacket, IN LPCXMEDIAPACKET pOutputPacket ) { HRESULT hr = S_OK; LPCXMEDIAPACKET pMediaPacket; if(m_fMicrophone) { RIP_ON_NOT_TRUE("Hawk Microphone XMediaObject::Process", NULL==pInputPacket); RIP_ON_NOT_TRUE("Hawk Microphone XMediaObject::Process", NULL!=pOutputPacket); RIP_ON_NOT_TRUE("Hawk Microphone XMediaObject::Process", pOutputPacket->dwMaxSize > (DWORD)m_pStreamingResources->BytesPerUSBFrame * HAWK_MIN_PACKET_FRAMES); RIP_ON_NOT_TRUE("Hawk Microphone XMediaObject::Process", 0 == pOutputPacket->dwMaxSize%HAWK_BYTES_PER_SAMPLE); pMediaPacket = pOutputPacket; } else { RIP_ON_NOT_TRUE("Hawk Headphone XMediaObject::Process", NULL!=pInputPacket); RIP_ON_NOT_TRUE("Hawk Headphone XMediaObject::Process", NULL==pOutputPacket); RIP_ON_NOT_TRUE("Hawk Headphone XMediaObject::Process", pInputPacket->dwMaxSize > (DWORD)m_pStreamingResources->BytesPerUSBFrame * HAWK_MIN_PACKET_FRAMES); RIP_ON_NOT_TRUE("Hawk Headphone XMediaObject::Process", 0 == pInputPacket->dwMaxSize%HAWK_BYTES_PER_SAMPLE); pMediaPacket = pInputPacket; } // Raise IRQL after rips KIRQL oldIrql = KeRaiseIrqlToDpcLevel(); if(m_fReady) { PHAWK_PACKET_CONTEXT pPacketContext = m_pStreamingResources->Free.RemoveHead(); if(pPacketContext) { // // Initialize the packet // XMOAcceptPacket(pMediaPacket); // // Copy the media packet into the hawk packet context. // The structure is identical except hawk has one extra // field, for queuing // RtlCopyMemory(&pPacketContext->MediaPacket, pMediaPacket, sizeof(XMEDIAPACKET)); // // Insert the packet onto the pending programming queue. // m_pStreamingResources->PendingProgram.InsertTail(pPacketContext); // // Jog the programming routine. // ProgramTransfer(); } else { hr = HRESULT_FROM_WIN32(ERROR_BUSY); } } else { hr = HRESULT_FROM_WIN32(ERROR_DEVICE_NOT_CONNECTED); } if(FAILED(hr)) { XMOCompletePacket(pMediaPacket, 0, m_pfnCallback, m_pvContext, XMEDIAPACKET_STATUS_FAILURE); } KeLowerIrql(oldIrql); return hr; } EXTERN_C HRESULT WINAPI XVoiceCreateMediaObjectInternal( IN PXPP_DEVICE_TYPE XppDeviceType, IN DWORD dwPort, IN DWORD dwMaxAttachedPackets, IN LPWAVEFORMATEX pwfxFormat, IN PFNXMEDIAOBJECTCALLBACK pfnCallback, IN PVOID pvContext, OUT XMediaObject **ppXmediaObject ) { HRESULT hr; XHawkMediaObject *pHawkMediaObject = NULL; hr = XVoiceCreateMediaObject( XppDeviceType, dwPort, dwMaxAttachedPackets, pwfxFormat, ppXmediaObject ); if (SUCCEEDED(hr)) { // // save callback and context // pHawkMediaObject = (XHawkMediaObject *) *ppXmediaObject; pHawkMediaObject->m_pvContext = pvContext; pHawkMediaObject->m_pfnCallback = pfnCallback; } return hr; } XBOXAPI EXTERN_C HRESULT WINAPI XVoiceCreateMediaObject( IN PXPP_DEVICE_TYPE XppDeviceType, IN DWORD dwPort, IN DWORD dwMaxAttachedPackets, IN PWAVEFORMATEX pwfxFormat OPTIONAL, OUT XMediaObject **ppXmediaObject ) /*++ Routine Description: This is the factory method for XHawkMediaObjects. It is an exported API for games. --*/ { // // We need the class Id to find the device. // KIRQL oldIrql; PNP_CLASS_ID classId; HRESULT hr = S_OK; XHawkMediaObject *pHawkMediaObject = NULL; UCHAR ucRateIndex=2; //default rate is 16 kHz, at index 2 in the rate table. // // Verify that XInitDevices has been called. // RIP_ON_NOT_TRUE_WITH_MESSAGE(XPP_XInitDevicesHasBeenCalled, "XVoiceCreateMediaObject: XInitDevices must be called first!"); // // Verify port // RIP_ON_NOT_TRUE(XVoiceCreateMediaObject, dwPort 1)); // // Use raised Irql to synchronize the allocation // and opening of instances // oldIrql = KeRaiseIrqlToDpcLevel(); // // Claim an existing instance of XHawkMediaObject // if(XDEVICE_TYPE_VOICE_MICROPHONE == XppDeviceType) { if(0==XHawkMediaObject::sm_AvailableMicrophoneResources) { KeLowerIrql(oldIrql); RIP("XVoiceCreateMediaObject: attempt to exceed number of mircophones requested in XInitDevices.\n"); return E_OUTOFMEMORY; } pHawkMediaObject = &XHawkMediaObject::sm_pMicrophones[dwPort]; } else if(XDEVICE_TYPE_VOICE_HEADPHONE == XppDeviceType) { if(0==XHawkMediaObject::sm_AvailableHeadphoneResources) { KeLowerIrql(oldIrql); RIP("XVoiceCreateMediaObject: attempt to exceed number of headphones requested in XInitDevices.\n"); return E_OUTOFMEMORY; } pHawkMediaObject = &XHawkMediaObject::sm_pHeadphones[dwPort]; } #if DBG else { KeLowerIrql(oldIrql); RIP("XVoiceCreateMediaObject: XppDeviceType is not a valid type for XVoiceCreateMediaObject\n"); } #endif //DBG // // Check the wave format // if(pwfxFormat) { // // Verify that the setting are supported // #if DBG KeLowerIrql(oldIrql); RIP_ON_NOT_TRUE(XVoiceCreateMediaObject, pwfxFormat->wFormatTag==WAVE_FORMAT_PCM); RIP_ON_NOT_TRUE(XVoiceCreateMediaObject, pwfxFormat->nChannels==1); RIP_ON_NOT_TRUE(XVoiceCreateMediaObject, 2*pwfxFormat->nSamplesPerSec==pwfxFormat->nAvgBytesPerSec); RIP_ON_NOT_TRUE(XVoiceCreateMediaObject, pwfxFormat->wBitsPerSample==16); RIP_ON_NOT_TRUE(XVoiceCreateMediaObject, pwfxFormat->nBlockAlign == 2); oldIrql = KeRaiseIrqlToDpcLevel(); #endif // // First validate rate against allowed rates. // for(ucRateIndex=0; ucRateIndex < HAWK_SAMPLE_RATE_COUNT; ucRateIndex++) { if(pwfxFormat->nSamplesPerSec == (DWORD)XHawkMediaObject::sm_HawkSampleRates[ucRateIndex].nSamplePerSecond) { break; } } #if DBG if(HAWK_SAMPLE_RATE_COUNT == ucRateIndex) { KeLowerIrql(oldIrql); RIP("XVoiceCreateMediaObject: pWaveFormatEx->nSamplesPerSec has illegal value"); } #endif //DBG } if( pHawkMediaObject->m_fOpened ) { pHawkMediaObject = NULL; hr = HRESULT_FROM_WIN32(ERROR_SHARING_VIOLATION); } else if (pHawkMediaObject->m_fConnected) { // // Set the sample rate // if(XHawkMediaObject::sm_pRateIndexRefCounts[dwPort]) // // If the reference count is non-zero, then the // other half of this device has set the rate // (microphone and headphone must have the same rate) // { XHawkMediaObject::sm_pRateIndexRefCounts[dwPort]++; // // Verify that the rates match // if(XHawkMediaObject::sm_pRateIndices[dwPort]!=ucRateIndex) { // Be kinder here, return an error, I could vaguely see this slipping through testing. KeLowerIrql(oldIrql); RIP("XVoiceCreateMediaObject: Attempt to open microphone and headphone of same device with different sampling rates."); return HRESULT_FROM_WIN32(ERROR_INVALID_PARAMETER); } } else // // The rate hasn't been set yet, so set it // { XHawkMediaObject::sm_pRateIndexRefCounts[dwPort] = 1; XHawkMediaObject::sm_pRateIndices[dwPort]=ucRateIndex; hr = pHawkMediaObject->SetSampleRate(ucRateIndex, oldIrql); hr = pHawkMediaObject->SetAGC(HAWK_AGC_ON, oldIrql); } // // Allocate the streaming resources. // if(SUCCEEDED(hr)) { hr = pHawkMediaObject->AllocateStreamingResources(dwMaxAttachedPackets, ucRateIndex); } if(SUCCEEDED(hr)) { pHawkMediaObject->m_fOpened = TRUE; pHawkMediaObject->m_fReady = TRUE; pHawkMediaObject->m_lRefCount = 1; } else { pHawkMediaObject = NULL; XHawkMediaObject::sm_pRateIndexRefCounts[dwPort]--; } } else { pHawkMediaObject = NULL; hr = HRESULT_FROM_WIN32(ERROR_DEVICE_NOT_CONNECTED); } // Record that we opened a device if(pHawkMediaObject) { if(pHawkMediaObject->m_fMicrophone) XHawkMediaObject::sm_AvailableMicrophoneResources--; else XHawkMediaObject::sm_AvailableHeadphoneResources--; } // // Restore Irql since the stuff that needs // to be synchronous is done. // KeLowerIrql(oldIrql); *ppXmediaObject = pHawkMediaObject; return hr; } HRESULT XHawkMediaObject::SetSampleRate(UCHAR ucRateIndex, KIRQL BaseIrql) /*++ Routine Description: Sends a command to hawk to change the sample rate. This routine opens the default pipe. Sends a vendor specific SetFeature to change the rate. The closes the default pipe. The routine is synchronous, which means it must drop to below DISPATCH_LEVEL. Parameters: ucRateIndex - rate index to send to the device. BaseIrql - the lowest irql we are permitted to drop to. Comments: Must be called at DPC level, BaseIrql must be lower than DPC. --*/ { USBD_STATUS usbdStatus; // // Open the default endpoints // URB Urb; USB_BUILD_OPEN_DEFAULT_ENDPOINT(&Urb.OpenEndpoint); usbdStatus = m_pDevice->SubmitRequest(&Urb); if(USBD_ERROR(usbdStatus)) { return HRESULT_FROM_WIN32(IUsbDevice::Win32FromUsbdStatus(usbdStatus)); } // // While still at DPC, set the m_fClosePending flag. This will prevent HawkRemoveDevice // from calling RemoveComplete on us. We take on the burden of checking for m_fRemovePending // when clearing this flag. // m_fClosePending = TRUE; // // Drop to BaseIrql so we can start doing the asynchronous parts synchronously. // ASSERT(BaseIrql < DISPATCH_LEVEL); KeLowerIrql(BaseIrql); // // Send the SET_FEATURE(SampleRate) command. // USB_BUILD_CONTROL_TRANSFER( &Urb.ControlTransfer, NULL, //Use the default endpoint NULL, //No transfer buffer 0, 0, NULL, //Do this transfer synchronously 0, FALSE, //Short Transfer NOT OK USB_VENDOR_COMMAND | USB_COMMAND_TO_INTERFACE | USB_HOST_TO_DEVICE, USB_REQUEST_SET_FEATURE, (USHORT) 0x0100 | ucRateIndex, // DD: ucRateIndex, //Rate Index to set, see Talon Specification. HAWK_VENDOR_FEATURE_SAMPLE_RATE, 0); //Transfer length is zero usbdStatus = m_pDevice->SubmitRequest(&Urb); // // Close the default endpoint // USB_BUILD_CLOSE_DEFAULT_ENDPOINT(&Urb.CloseEndpoint, NULL, NULL); m_pDevice->SubmitRequest(&Urb); // // Now all the synchronous stuff is done we can // go back up to DISPATCH_LEVEL // KeRaiseIrqlToDpcLevel(); // // We are not really open yet, nor are we pending a close. // m_fOpened = FALSE; m_fClosePending = FALSE; // // Check the remove pending flag, and complete the remove // it one was pending. // Also fail this request with ERROR_DEVICE_NOT_CONNECTED. // if(m_fRemovePending) { m_pDevice->SetExtension(NULL); m_pDevice->RemoveComplete(); m_pDevice = NULL; m_fRemovePending = FALSE; return HRESULT_FROM_WIN32(ERROR_DEVICE_NOT_CONNECTED); } // // If we are supporting fixed rate devices // tolerate a failure, if the // requested rate was 16 ksamples/second #ifdef HAWK_SUPPORT_FIXED_RATE if(ucRateIndex==2) { usbdStatus = USBD_STATUS_SUCCESS; } #endif //HAWK_SUPPORT_FIXED_RATE // // Return the status of the request to set the data rate. // return HRESULT_FROM_WIN32(IUsbDevice::Win32FromUsbdStatus(usbdStatus)); } HRESULT XHawkMediaObject::SetAGC(UCHAR ucAGC, KIRQL BaseIrql) /*++ Routine Description: Sends a command to hawk to change the sample rate. This routine opens the default pipe. Sends a vendor specific SetFeature to change the rate. The closes the default pipe. The routine is synchronous, which means it must drop to below DISPATCH_LEVEL. Parameters: ucAGC - rate index to send to the device. BaseIrql - the lowest irql we are permitted to drop to. Comments: Must be called at DPC level, BaseIrql must be lower than DPC. --*/ { USBD_STATUS usbdStatus; // // Open the default endpoints // URB Urb; USB_BUILD_OPEN_DEFAULT_ENDPOINT(&Urb.OpenEndpoint); usbdStatus = m_pDevice->SubmitRequest(&Urb); if(USBD_ERROR(usbdStatus)) { return HRESULT_FROM_WIN32(IUsbDevice::Win32FromUsbdStatus(usbdStatus)); } // // While still at DPC, set the m_fClosePending flag. This will prevent HawkRemoveDevice // from calling RemoveComplete on us. We take on the burden of checking for m_fRemovePending // when clearing this flag. // m_fClosePending = TRUE; // // Drop to BaseIrql so we can start doing the asynchronous parts synchronously. // ASSERT(BaseIrql < DISPATCH_LEVEL); KeLowerIrql(BaseIrql); // // Send the SET_FEATURE(SampleRate) command. // USB_BUILD_CONTROL_TRANSFER( &Urb.ControlTransfer, NULL, //Use the default endpoint NULL, //No transfer buffer 0, 0, NULL, //Do this transfer synchronously 0, FALSE, //Short Transfer NOT OK USB_VENDOR_COMMAND | USB_COMMAND_TO_INTERFACE | USB_HOST_TO_DEVICE, USB_REQUEST_SET_FEATURE, (USHORT)ucAGC, //ON/OFF, see Talon Specification. (USHORT)HAWK_VENDOR_FEATURE_AGC, 0); //Transfer length is zero usbdStatus = m_pDevice->SubmitRequest(&Urb); // // Close the default endpoint // USB_BUILD_CLOSE_DEFAULT_ENDPOINT(&Urb.CloseEndpoint, NULL, NULL); m_pDevice->SubmitRequest(&Urb); // // Now all the synchronous stuff is done we can // go back up to DISPATCH_LEVEL // KeRaiseIrqlToDpcLevel(); // // We are not really open yet, nor are we pending a close. // m_fOpened = FALSE; m_fClosePending = FALSE; // // Check the remove pending flag, and complete the remove // it one was pending. // Also fail this request with ERROR_DEVICE_NOT_CONNECTED. // if(m_fRemovePending) { m_pDevice->SetExtension(NULL); m_pDevice->RemoveComplete(); m_pDevice = NULL; m_fRemovePending = FALSE; return HRESULT_FROM_WIN32(ERROR_DEVICE_NOT_CONNECTED); } // // Return the status of the request to set the data rate. // return HRESULT_FROM_WIN32(IUsbDevice::Win32FromUsbdStatus(usbdStatus)); } HRESULT XHawkMediaObject::AllocateStreamingResources(ULONG ulMaxAttachedPackets, ULONG ulRateIndex) /*++ Routine Description: This is a helper function called by Factory Method, XVoiceCreateMediaObject, to allocate resources for providing streaming. --*/ { ULONG i; // // Allocate memory for Attached Packets // PHAWK_PACKET_CONTEXT pPacketContextArray = new HAWK_PACKET_CONTEXT[ulMaxAttachedPackets]; if(!pPacketContextArray) return E_OUTOFMEMORY; RtlZeroMemory(pPacketContextArray, sizeof(HAWK_PACKET_CONTEXT)*ulMaxAttachedPackets); // // Grab a free streaming resources (there has to be one // at this point or there is a bug somewhere in this driver.) // PHAWK_STREAMING_RESOURCES pResources = sm_pFreeStreamingResources; ASSERT(pResources); sm_pFreeStreamingResources = pResources->pNextFree; RtlZeroMemory(pResources, sizeof(HAWK_STREAMING_RESOURCES)); // // Populate the free list of packet contexts // pResources->m_pContextArray = pPacketContextArray; for(i=0; i < ulMaxAttachedPackets; i++) { pResources->Free.InsertTail(pPacketContextArray + i); } // // Copy the sample rate parameters from the rate table into the // pResources structure for quick access. // pResources->BytesPerUSBFrame = sm_HawkSampleRates[ulRateIndex].BytesPerUSBFrame; pResources->ExtraSampleInterval = sm_HawkSampleRates[ulRateIndex].ExtraSampleInterval; pResources->AvailableTransferContexts = HAWK_TRANSFER_CONTEXT_BOTH; //pResources->ProgrammedFrame = 0; //Not necessary since the memory was zeroed. pResources->TransferContext[0].pHawkMediaObject = pResources->TransferContext[1].pHawkMediaObject = this; pResources->TransferContext[0].ucContextNumber = HAWK_TRANSFER_CONTEXT_0; pResources->TransferContext[1].ucContextNumber = HAWK_TRANSFER_CONTEXT_1; // // Open the isoch endpoint // URB_ISOCH_OPEN_ENDPOINT OpenUrb; USB_BUILD_ISOCH_OPEN_ENDPOINT( &OpenUrb, m_bEndpointAddress, m_wMaxPacket, 0); USBD_STATUS status = m_pDevice->SubmitRequest((PURB)&OpenUrb); if(USBD_ERROR(status)) { delete [] pPacketContextArray; pResources->pNextFree = sm_pFreeStreamingResources; sm_pFreeStreamingResources = pResources; return HRESULT_FROM_WIN32(IUsbDevice::Win32FromUsbdStatus(status)); } pResources->EndpointHandle = OpenUrb.EndpointHandle; // // start the stream (since we don't have a start API) // URB_ISOCH_START_TRANSFER StartTransferUrb; USB_BUILD_ISOCH_START_TRANSFER( &StartTransferUrb, pResources->EndpointHandle, 0, URB_FLAG_ISOCH_START_ASAP ); status = m_pDevice->SubmitRequest((PURB)&StartTransferUrb); ASSERT(USBD_SUCCESS(status)); // // The stream should be ready to go, we just need to // attach transfers. // m_pStreamingResources = pResources; return S_OK; } void XHawkMediaObject::CloseEndpoint( void ) { ASSERT(m_pStreamingResources); ASSERT(m_pStreamingResources->EndpointHandle); // // The device is not ready, if the endpoint is not open. // m_fReady = FALSE; if(m_fClosingEndpoint) return; m_fClosingEndpoint = TRUE; // // Build and submit the close URB for the Isoch endpoint // USB_BUILD_ISOCH_CLOSE_ENDPOINT( &m_pStreamingResources->CloseUrb, m_pStreamingResources->EndpointHandle, (PURB_COMPLETE_PROC)&XHawkMediaObject::CloseEndpointComplete, (PVOID)this ); m_pDevice->SubmitRequest((PURB)&m_pStreamingResources->CloseUrb); } void XHawkMediaObject::CloseEndpointComplete (PURB pUrb, XHawkMediaObject *pThis) /*++ Routine Description: This is the call back when an isoch endpoint is closed. This could happen if a close is pending, or if a remove is pending, or both. --*/ { DWORD dwPort; ASSERT(USBD_SUCCESS(pUrb->Header.Status)); // // Now mark the endpoint closed. // pThis->m_pStreamingResources->EndpointHandle = NULL; // // Decrement the reference count on the rate, for // which we need the port index. // if(pThis->m_fMicrophone) { dwPort = (DWORD)(pThis - XHawkMediaObject::sm_pMicrophones); } else { dwPort = (DWORD)(pThis - XHawkMediaObject::sm_pHeadphones); } XHawkMediaObject::sm_pRateIndexRefCounts[dwPort]--; // // Abort outstanding packets. // pThis->AbortMediaPackets(); // // Deal with the remove pending case. // if(pThis->m_fRemovePending) { pThis->m_pDevice->SetExtension(NULL); pThis->m_pDevice->RemoveComplete(); pThis->m_pDevice = NULL; pThis->m_fRemovePending = FALSE; } // // If this is the close pending case, // then there is an event that we need to set // if(pThis->m_fClosePending) { // // We have to clear the m_fClosePending // and m_fOpened flags while still at // DPC for synchronization purposes. // pThis->m_fClosePending = FALSE; pThis->m_fOpened = FALSE; // // Signal the close event so that // FreeStreamingResources can continue // to clean up. // KeSetEvent( &pThis->m_pStreamingResources->CloseEvent, IO_NO_INCREMENT, FALSE ); } pThis->m_fClosingEndpoint = FALSE; } void XHawkMediaObject::FreeStreamingResources (void) { KIRQL oldIrql = KeRaiseIrqlToDpcLevel(); if(m_fConnected && m_pStreamingResources->EndpointHandle) { // // Mark that this object is pending close // and initialize the close event. // m_fClosePending = TRUE; KeInitializeEvent( &m_pStreamingResources->CloseEvent, SynchronizationEvent, FALSE); // // If it is already pending remove // then we can just wait for // the endpoint to close, other // wise we must initiate closing // it. // if(!m_fRemovePending) { CloseEndpoint(); } // // Lower Irql and let the endpoint close. // KeLowerIrql(oldIrql); // // The endpoint close is under way, // wait for it to complete. // KeWaitForSingleObject ( &m_pStreamingResources->CloseEvent, Executive, KernelMode, FALSE, NULL ); } else { m_fOpened = FALSE; KeLowerIrql(oldIrql); } if(m_fMicrophone) sm_AvailableMicrophoneResources++; else sm_AvailableHeadphoneResources++; // // Now we can cleanup what is left of the streaming resources // delete [] m_pStreamingResources->m_pContextArray; m_pStreamingResources->pNextFree = sm_pFreeStreamingResources; sm_pFreeStreamingResources = m_pStreamingResources; m_pStreamingResources = NULL; } void XHawkMediaObject::ProgramTransfer() /*++ Routine Description: This routine translates a portion of a packet into USB Isochronous Transfer Requests. Note: Sometimes this routine in called at PASSIVE_LEVEL and sometimes at DISPATCH_LEVEL. This routine is not inherently reentrant so we raise the IRQL to DISPATCH_LEVEL all the time. --*/ { KIRQL oldIrql = KeRaiseIrqlToDpcLevel(); USBD_ISOCH_BUFFER_DESCRIPTOR isochBufferDescriptor; PHAWK_TRANSFER_CONTEXT pTransferContext = NULL; int i; // // Claim a context // if(TEST_FLAG(m_pStreamingResources->AvailableTransferContexts, HAWK_TRANSFER_CONTEXT_0)) { CLEAR_FLAG(m_pStreamingResources->AvailableTransferContexts, HAWK_TRANSFER_CONTEXT_0); pTransferContext = &m_pStreamingResources->TransferContext[0]; } else if(TEST_FLAG(m_pStreamingResources->AvailableTransferContexts, HAWK_TRANSFER_CONTEXT_1)) { CLEAR_FLAG(m_pStreamingResources->AvailableTransferContexts, HAWK_TRANSFER_CONTEXT_1); pTransferContext = &m_pStreamingResources->TransferContext[1]; } isochBufferDescriptor.TransferComplete = (PFNUSBD_ISOCH_TRANSFER_COMPLETE)XHawkMediaObject::TransferComplete; while(pTransferContext) { PHAWK_PACKET_CONTEXT pPacketContext = m_pStreamingResources->PendingProgram.GetHead(); // // If there are no more packet contexts pending programming, break the // loop, we are done. // if(!pPacketContext) { SET_FLAG(m_pStreamingResources->AvailableTransferContexts, pTransferContext->ucContextNumber); break; } // // Fill out the transfer context // pTransferContext->pPacketContext = pPacketContext; pTransferContext->pucTransferBuffer = (PUCHAR) pPacketContext->MediaPacket.pvBuffer; pTransferContext->ulTransferBytes = 0; pTransferContext->cFirstFrame = m_pStreamingResources->ProgrammedFrame; pTransferContext->fWrapBuffer = FALSE; pTransferContext->fLastTransfer = FALSE; // // Walk through each frame, up to HAWK_USB_MAX_FRAMES, // for(i=0; i < HAWK_USB_MAX_FRAMES; i++) { isochBufferDescriptor.Pattern[i] = m_pStreamingResources->BytesPerUSBFrame; if( m_pStreamingResources->ExtraSampleInterval && (++m_pStreamingResources->ProgrammedFrame == m_pStreamingResources->ExtraSampleInterval) ) { isochBufferDescriptor.Pattern[i] += 2; m_pStreamingResources->ProgrammedFrame=0; } pTransferContext->ulTransferBytes += isochBufferDescriptor.Pattern[i]; if(pTransferContext->ulTransferBytes > pPacketContext->MediaPacket.dwMaxSize) { m_pStreamingResources->WrapBuffer.ulSecondPacketBytes = pTransferContext->ulTransferBytes - pPacketContext->MediaPacket.dwMaxSize; pTransferContext->ulTransferBytes = pPacketContext->MediaPacket.dwMaxSize; pTransferContext->fWrapBuffer = TRUE; // // If this is for a headphones copy the output bytes // for this packet. // if(!m_fMicrophone) { RtlCopyMemory( m_pStreamingResources->WrapBuffer.Buffer, pPacketContext->MediaPacket.pvBuffer, pTransferContext->ulTransferBytes); } } if(pTransferContext->ulTransferBytes == pPacketContext->MediaPacket.dwMaxSize) { pTransferContext->fLastTransfer = TRUE; i++; break; } } ASSERT(i==8 || pTransferContext->fLastTransfer); pPacketContext->MediaPacket.dwMaxSize -= pTransferContext->ulTransferBytes; isochBufferDescriptor.Context = pTransferContext; isochBufferDescriptor.FrameCount = i; isochBufferDescriptor.TransferBuffer = pPacketContext->MediaPacket.pvBuffer; pPacketContext->MediaPacket.pvBuffer = ((PUCHAR)pPacketContext->MediaPacket.pvBuffer) + pTransferContext->ulTransferBytes; // // If we are done with the current packet, then pop it off the pending program queue. // if(pTransferContext->fLastTransfer) { m_pStreamingResources->PendingProgram.RemoveHead(); } // // If the wrap buffer is in use than we need to get the next // packet and start programming it in the beginning of the wrap buffer. // if(pTransferContext->fWrapBuffer) { DBGPRINT(1,("W")); PHAWK_WRAP_BUFFER pWrapBuffer = &m_pStreamingResources->WrapBuffer; isochBufferDescriptor.TransferBuffer = pWrapBuffer->Buffer; pPacketContext = m_pStreamingResources->PendingProgram.GetHead(); if(pPacketContext) { pWrapBuffer->pSecondPacket = pPacketContext; pWrapBuffer->pucTransferBuffer = (PUCHAR)pPacketContext->MediaPacket.pvBuffer; ASSERT(pWrapBuffer->ulSecondPacketBytes <= pPacketContext->MediaPacket.dwMaxSize); if(!m_fMicrophone) { RtlCopyMemory( pWrapBuffer->Buffer+pTransferContext->ulTransferBytes, pWrapBuffer->pucTransferBuffer, pWrapBuffer->ulSecondPacketBytes); } pPacketContext->MediaPacket.pvBuffer = pWrapBuffer->pucTransferBuffer + pWrapBuffer->ulSecondPacketBytes; pPacketContext->MediaPacket.dwMaxSize -= pWrapBuffer->ulSecondPacketBytes; } else // // No more packets pending // { pWrapBuffer->pSecondPacket = NULL; pWrapBuffer->pucTransferBuffer = NULL; if(!m_fMicrophone) { RtlZeroMemory( pWrapBuffer->Buffer+pTransferContext->ulTransferBytes, pWrapBuffer->ulSecondPacketBytes); } } } // // Build and submit a USB request // URB_ISOCH_ATTACH_BUFFER Urb; USB_BUILD_ISOCH_ATTACH_BUFFER( &Urb, m_pStreamingResources->EndpointHandle, USBD_DELAY_INTERRUPT_0_MS, &isochBufferDescriptor ); USBD_STATUS usbdStatus = m_pDevice->SubmitRequest((PURB)&Urb); ASSERT(USBD_SUCCESS(usbdStatus)); // // Try to claim another context // if(TEST_FLAG(m_pStreamingResources->AvailableTransferContexts, HAWK_TRANSFER_CONTEXT_0)) { CLEAR_FLAG(m_pStreamingResources->AvailableTransferContexts, HAWK_TRANSFER_CONTEXT_0); pTransferContext = &m_pStreamingResources->TransferContext[0]; } else if(TEST_FLAG(m_pStreamingResources->AvailableTransferContexts, HAWK_TRANSFER_CONTEXT_1)) { CLEAR_FLAG(m_pStreamingResources->AvailableTransferContexts, HAWK_TRANSFER_CONTEXT_1); pTransferContext = &m_pStreamingResources->TransferContext[1]; } else { // // no more contexts // pTransferContext = NULL; } } KeLowerIrql(oldIrql); } void XHawkMediaObject::TransferComplete(PUSBD_ISOCH_TRANSFER_STATUS Status, PVOID pvContext) /*++ Routine Description: This is the completion routine whenever an isoch transfer completes. It continues to program the current media packet, releases the media packet when it is complete, and moves on to the next packet. --*/ { PHAWK_TRANSFER_CONTEXT pTransferContext = (PHAWK_TRANSFER_CONTEXT)pvContext; XHawkMediaObject *pThis = pTransferContext->pHawkMediaObject; PHAWK_STREAMING_RESOURCES pStreamingResources = pThis->m_pStreamingResources; PHAWK_PACKET_CONTEXT pPacketContext = pTransferContext->pPacketContext; CHAR extraSampleClock = pThis->m_pStreamingResources->ExtraSampleInterval; if(extraSampleClock) extraSampleClock--; // // Post processing of data is only useful\needed for microphones. // if(pThis->m_fMicrophone) { ULONG bytesThisFrame; PUSHORT pBufferPosition; if(pTransferContext->fWrapBuffer) { pBufferPosition = (PUSHORT)pStreamingResources->WrapBuffer.Buffer; } else { pBufferPosition = (PUSHORT)pTransferContext->pucTransferBuffer; } // // First handle overflow and underflow conditions // for(ULONG ulFrameIndex = 0; ulFrameIndex < Status->FrameCount; ulFrameIndex++) { bytesThisFrame = pStreamingResources->BytesPerUSBFrame; if( extraSampleClock && (ulFrameIndex + pTransferContext->cFirstFrame == (ULONG)extraSampleClock) ) { bytesThisFrame += 2; } // // Fill in short frames with data copied from the previous samples(s) // if(USBD_ISOCH_STATUS_DATA_UNDERRUN == Status->PacketStatus[ulFrameIndex].ConditionCode) { //Copy bytes from the previous sample if(Status->PacketStatus[ulFrameIndex].BytesRead) { #if DBG if(0!=Status->PacketStatus[ulFrameIndex].BytesRead%2) { DBGPRINT(1,("Hardware violated sample boundary")); } #endif ULONG samplesToDuplicate = (bytesThisFrame - Status->PacketStatus[ulFrameIndex].BytesRead) >> 1; pBufferPosition = (PUSHORT)(((ULONG_PTR)pBufferPosition) + Status->PacketStatus[ulFrameIndex].BytesRead-2); USHORT sample = *pBufferPosition++; while(samplesToDuplicate--) { *pBufferPosition++ = sample; } } else // // This handles devices that mute by sending zero length data packets. // { RtlZeroMemory(pBufferPosition,bytesThisFrame); pBufferPosition = (PUSHORT)((ULONG_PTR)pBufferPosition + bytesThisFrame); } } else { pBufferPosition = (PUSHORT)((ULONG_PTR)pBufferPosition + bytesThisFrame); } // // Check for data overrun to try to adapt to // small clock differences // if( extraSampleClock && (USBD_ISOCH_STATUS_DATA_OVERRUN == Status->PacketStatus[ulFrameIndex].ConditionCode) ) { CHAR clockDifference = extraSampleClock - (pTransferContext->cFirstFrame+(CHAR)ulFrameIndex); pStreamingResources->ProgrammedFrame += clockDifference; } } // // If the wrap buffer was in use, then we need to copy the bytes back to their // original buffer // if(pTransferContext->fWrapBuffer) { // // First copy the part for the first packet // RtlCopyMemory( pTransferContext->pucTransferBuffer, pStreamingResources->WrapBuffer.Buffer, pTransferContext->ulTransferBytes ); // // Then copy the part for the second packet // if(pStreamingResources->WrapBuffer.pSecondPacket) { RtlCopyMemory( pStreamingResources->WrapBuffer.pucTransferBuffer, pStreamingResources->WrapBuffer.Buffer + pTransferContext->ulTransferBytes, pStreamingResources->WrapBuffer.ulSecondPacketBytes ); } } } else { #if DBG for(ULONG ulFrameIndex = 0; ulFrameIndex < Status->FrameCount; ulFrameIndex++) { if(Status->PacketStatus[ulFrameIndex].ConditionCode!=USBD_STATUS_SUCCESS) { DBGPRINT(1,("Failed Write: Frame=%d, CC=0x%0.1x, BytesWritten=0x%0.3x", ulFrameIndex, (DWORD)Status->PacketStatus[ulFrameIndex].ConditionCode, (DWORD)Status->PacketStatus[ulFrameIndex].BytesRead )); } } #endif } // // Post processing of data is complete, update completed size // if(pPacketContext->MediaPacket.pdwCompletedSize) *pPacketContext->MediaPacket.pdwCompletedSize += pTransferContext->ulTransferBytes; if( pTransferContext->fWrapBuffer&& pStreamingResources->WrapBuffer.pSecondPacket&& pStreamingResources->WrapBuffer.pSecondPacket->MediaPacket.pdwCompletedSize ) { *pStreamingResources->WrapBuffer.pSecondPacket->MediaPacket.pdwCompletedSize += pStreamingResources->WrapBuffer.ulSecondPacketBytes; } // // Mark the pTransferContext as unused // BOOL fLastTransfer = pTransferContext->fLastTransfer; SET_FLAG( pStreamingResources->AvailableTransferContexts, pTransferContext->ucContextNumber ); // // Complete packet if necessary // if(fLastTransfer) { // // Cache the media packet info on the stack // so we can free the packet context // before calling XMOCompletePacket. // XMEDIAPACKET mediaPacket; DWORD dwCompletedSize = 0; RtlCopyMemory(&mediaPacket, &pPacketContext->MediaPacket, sizeof(XMEDIAPACKET)); if(mediaPacket.pdwCompletedSize) dwCompletedSize = *mediaPacket.pdwCompletedSize; // // Place the packet back on the free list (do this before // calling XMOCompletePacket) so that a packet context // is available during the callback (if a callaback is // used) to program the next packet. // pStreamingResources->Free.InsertTail(pPacketContext); XMOCompletePacket(&mediaPacket, dwCompletedSize, pThis->m_pfnCallback, pThis->m_pvContext, XMEDIAPACKET_STATUS_SUCCESS); } // // Jog the program loop (if the device is still opened and ready). // if(pThis->m_fReady) { pThis->ProgramTransfer(); } } void XHawkMediaObject::AbortMediaPackets() { PHAWK_PACKET_CONTEXT pPacketContext = NULL; if(!TEST_FLAG(m_pStreamingResources->AvailableTransferContexts,HAWK_TRANSFER_CONTEXT_0)) { // // Complete the packet (it cannot be on the pending program list) // if(m_pStreamingResources->TransferContext[0].fLastTransfer) { pPacketContext = m_pStreamingResources->TransferContext[0].pPacketContext; } } if((NULL==pPacketContext )&& !TEST_FLAG(m_pStreamingResources->AvailableTransferContexts,HAWK_TRANSFER_CONTEXT_1)) { // // Complete the packet (it cannot be on the pending program list) // if(m_pStreamingResources->TransferContext[1].fLastTransfer) { pPacketContext = m_pStreamingResources->TransferContext[1].pPacketContext; } } if(NULL==pPacketContext) { pPacketContext = m_pStreamingResources->PendingProgram.RemoveHead(); } while(pPacketContext) { // // Cache the media packet info on the stack // so we can free the packet context // before calling XMOCompletePacket. // XMEDIAPACKET mediaPacket; RtlCopyMemory(&mediaPacket, &pPacketContext->MediaPacket, sizeof(XMEDIAPACKET)); // // Place the packet back on the free list (do this before // calling XMOCompletePacket) so that a packet context // is available during the callback (if a callaback is // used) to program the next packet. // m_pStreamingResources->Free.InsertTail(pPacketContext); XMOCompletePacket(&mediaPacket, 0, m_pfnCallback, m_pvContext, XMEDIAPACKET_STATUS_FLUSHED); // // Get the next packet. // pPacketContext = m_pStreamingResources->PendingProgram.RemoveHead(); } } //------------------------------------------------------------------------- // CHawkPacketQueue implementation //------------------------------------------------------------------------- void CHawkPacketQueue::InsertTail(PHAWK_PACKET_CONTEXT pPacketContext) { pPacketContext->pNextPacket = NULL; if(m_pTail) { m_pTail->pNextPacket = pPacketContext; } else { m_pHead = pPacketContext; } m_pTail = pPacketContext; } PHAWK_PACKET_CONTEXT CHawkPacketQueue::RemoveHead() { PHAWK_PACKET_CONTEXT pRetVal; if(m_pHead) { pRetVal = m_pHead; m_pHead = m_pHead->pNextPacket; if(!m_pHead) m_pTail = NULL; } else { pRetVal = NULL; } return pRetVal; }