|
|
//==========================================================================;
//
// THIS CODE AND INFORMATION IS PROVIDED "AS IS" WITHOUT WARRANTY OF ANY
// KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
// IMPLIED WARRANTIES OF MERCHANTABILITY AND/OR FITNESS FOR A PARTICULAR
// PURPOSE.
//
// Copyright (c) 1992 - 1996 Microsoft Corporation. All Rights Reserved.
//
//==========================================================================;
#include <strmini.h>
#include <ksmedia.h>
#include "kskludge.h"
#include "codmain.h"
#include "coddebug.h"
#include <ntstatus.h>
#include "defaults.h"
#include "ccdecode.h"
#include "ccformatcodes.h"
#ifdef PERFTEST
extern enum STREAM_DEBUG_LEVEL _CDebugLevel; enum STREAM_DEBUG_LEVEL OldLevel; ULONGLONG PerfThreshold = 250; #endif // PERFTEST
//==========================================================================;
// Routines for processing VBI streams
//==========================================================================;
void CheckResultsArray( PHW_DEVICE_EXTENSION pHwDevExt, unsigned int StartLine, unsigned int EndLine ) { PDSPRESULT new;
//
// (Re)size the results array if needed
//
if( 0 == pHwDevExt->DSPResult || EndLine > pHwDevExt->DSPResultEndLine || StartLine < pHwDevExt->DSPResultStartLine ) { if (StartLine > pHwDevExt->DSPResultStartLine) StartLine = pHwDevExt->DSPResultStartLine;
if (EndLine < pHwDevExt->DSPResultEndLine) EndLine = pHwDevExt->DSPResultEndLine;
new = ( PDSPRESULT ) ExAllocatePool( NonPagedPool, sizeof( DSPRESULT ) * ( EndLine - StartLine + 1 ) );
if( new ) { if (pHwDevExt->DSPResult) ExFreePool( pHwDevExt->DSPResult ); pHwDevExt->DSPResult = new; pHwDevExt->DSPResultStartLine = StartLine; pHwDevExt->DSPResultEndLine = EndLine;
CDebugPrint( DebugLevelInfo, (CODECNAME ": Resized results array\n" )); } else { CDebugPrint( DebugLevelInfo, (CODECNAME ": Resize results array failed\n" )); CASSERT( new ); pHwDevExt->Statistics.Common.InternalErrors++; } } }
/*
** CheckNewVBIInfo ** ** Checks for a new VBIInfoHeader ** ** Here's a little trickery to save having separate builds for the infinite pin ** tee and MSTee filters. IPT, being Ring3, doesn't pass VBIInfoHeaders, but it ** does pass the flags to show they've changed. We only make a copy if the ** data is good, otherwise we stick with the default header we started with. ** ** Arguments: ** ** PHW_DEVICE_EXTENSION pHwDevExt ** PSTREAMEX pInStrmEx ** PKS_VBI_FRAME_INFO pInVBIFrameInfo ** PKSSTREAM_HEADER pInStreamHeader ** ** Returns: nothing ** ** Side Effects: none */ int CheckNewVBIInfo( PHW_DEVICE_EXTENSION pHwDevExt, PSTREAMEX pInStrmEx, PKS_VBI_FRAME_INFO pInVBIFrameInfo ) { PKS_VBIINFOHEADER pVBIInfoHeader = &pInStrmEx->CurrentVBIInfoHeader; PVBICODECFILTERING_STATISTICS_CC Stats = &pHwDevExt->Statistics;
if( 0 == pInVBIFrameInfo->VBIInfoHeader.StartLine || 0 == pInVBIFrameInfo->VBIInfoHeader.EndLine || 0 == pInVBIFrameInfo->VBIInfoHeader.ActualLineStartTime ) { return 0; }
CDebugPrint( DebugLevelInfo, (CODECNAME ": VBIInfoHeader Change\n" )); Stats->Common.VBIHeaderChanges++;
//
// Resize the results array if needed
//
CheckResultsArray(pHwDevExt, pInVBIFrameInfo->VBIInfoHeader.StartLine, pInVBIFrameInfo->VBIInfoHeader.EndLine);
//
// Copy new VBI info header over old
//
RtlCopyMemory( pVBIInfoHeader, &pInVBIFrameInfo->VBIInfoHeader, sizeof( KS_VBIINFOHEADER )); //pVBIInfoHeader->ActualLineStartTime = 780;
RtlZeroMemory( &pInStrmEx->ScanlinesDiscovered, sizeof( pInStrmEx->ScanlinesDiscovered )); RtlZeroMemory( &pInStrmEx->SubstreamsDiscovered, sizeof( pInStrmEx->SubstreamsDiscovered )); CDebugPrint( DebugLevelVerbose, ( CODECNAME ": VBIInfoHeader->StartLine %lu\n", pVBIInfoHeader->StartLine )); CDebugPrint( DebugLevelVerbose, ( CODECNAME ": VBIInfoHeader->EndLine %lu\n", pVBIInfoHeader->EndLine )); //CDebugPrint( DebugLevelVerbose,
// ( CODECNAME ": VBIInfoHeader->SamplingFrequency %lu\n",
// pVBIInfoHeader->SamplingFrequency ));
CDebugPrint( DebugLevelVerbose, ( CODECNAME ": VBIInfoHeader->MinLineStartTime %lu\n", pVBIInfoHeader->MinLineStartTime )); CDebugPrint( DebugLevelVerbose, ( CODECNAME ": VBIInfoHeader->MaxLineStartTime %lu\n", pVBIInfoHeader->MaxLineStartTime )); CDebugPrint( DebugLevelVerbose, ( CODECNAME ": VBIInfoHeader->ActualLineStartTime %lu\n", pVBIInfoHeader->ActualLineStartTime )); CDebugPrint( DebugLevelVerbose, ( CODECNAME ": VBIInfoHeader->ActualLineEndTime %lu\n", pVBIInfoHeader->ActualLineEndTime )); CDebugPrint( DebugLevelVerbose, ( CODECNAME ": VBIInfoHeader->VideoStandard %lu\n", pVBIInfoHeader->VideoStandard )); CDebugPrint( DebugLevelVerbose, ( CODECNAME ": VBIInfoHeader->SamplesPerLine %lu\n", pVBIInfoHeader->SamplesPerLine )); CDebugPrint( DebugLevelVerbose, ( CODECNAME ": VBIInfoHeader->StrideInBytes %lu\n", pVBIInfoHeader->StrideInBytes )); CDebugPrint( DebugLevelVerbose, ( CODECNAME ": VBIInfoHeader->BufferSize %lu\n", pVBIInfoHeader->BufferSize ));
return 1; }
/*
** ProcessChannelChange ** ** Handles a VBI_FLAG_TVTUNER_CHANGE event ** ** Arguments: ** ** PHW_DEVICE_EXTENSION pHwDevExt ** PSTREAMEX pInStrmEx ** PKS_VBI_FRAME_INFO pInVBIFrameInfo ** PKSSTREAM_HEADER pInStreamHeader ** ** Returns: nothing ** ** Side Effects: none */ int ProcessChannelChange( PHW_DEVICE_EXTENSION pHwDevExt, PSTREAMEX pInStrmEx, PKS_VBI_FRAME_INFO pInVBIFrameInfo, PKSSTREAM_HEADER pInStreamHeader ) { PKS_VBIINFOHEADER pVBIInfoHeader = &pInStrmEx->CurrentVBIInfoHeader; PVBICODECFILTERING_STATISTICS_CC Stats = &pHwDevExt->Statistics; PKS_TVTUNER_CHANGE_INFO pChangeInfo = &pInVBIFrameInfo->TvTunerChangeInfo; ULONG CurrentStrmEx; ULONG i;
if( pChangeInfo->dwFlags & KS_TVTUNER_CHANGE_BEGIN_TUNE ) { CDebugPrint( DebugLevelInfo, (CODECNAME ": TVTuner Change START\n" )); pHwDevExt->fTunerChange = TRUE; } else if( pChangeInfo->dwFlags & KS_TVTUNER_CHANGE_END_TUNE ) { Stats->Common.TvTunerChanges++; pHwDevExt->fTunerChange = FALSE; CDebugPrint( DebugLevelInfo, (CODECNAME ": TVTuner Change END\n" )); RtlZeroMemory( &pInStrmEx->ScanlinesDiscovered, sizeof( pInStrmEx->ScanlinesDiscovered )); RtlZeroMemory( &pInStrmEx->SubstreamsDiscovered, sizeof( pInStrmEx->SubstreamsDiscovered )); CurrentStrmEx = 0; //
// Flag a discontuity. This is passed to outgoing SRBs and will force
// the downstream Line 21 decoder to clear its current CC data off the
// screen.
//
for( i = 0; i < pHwDevExt->ActualInstances[STREAM_CC]; i++ ) { PSTREAMEX pOutStrmEx; PHW_STREAM_REQUEST_BLOCK pOutSrb; PVBICODECFILTERING_STATISTICS_CC_PIN PinStats; do { CASSERT( CurrentStrmEx < MAX_PIN_INSTANCES ); pOutStrmEx = pHwDevExt->pStrmEx[STREAM_CC][CurrentStrmEx++]; }while( !pOutStrmEx ); PinStats = &pOutStrmEx->PinStats; //
// Get the next output stream SRB if it's available.
//
if( QueueRemove( &pOutSrb, &pOutStrmEx->StreamDataSpinLock, &pOutStrmEx->StreamDataQueue ) ) { PKSSTREAM_HEADER pOutStreamHeader = pOutSrb->CommandData.DataBufferArray; PKS_VBI_FRAME_INFO pOutVBIFrameInfo = (PKS_VBI_FRAME_INFO)(pOutStreamHeader+1); PUCHAR pOutData = (PUCHAR)pOutStreamHeader->Data;
if( pOutStreamHeader->FrameExtent < CCSamples ) { CDebugPrint( DebugLevelError, ( CODECNAME ": Outgoing Data SRB buffer is too small %u\n", pOutStreamHeader->FrameExtent )); PinStats->Common.InternalErrors++; Stats->Common.OutputFailures++; pOutStreamHeader->DataUsed = 0; } else { PinStats->Common.SRBsProcessed++; Stats->Common.OutputSRBsProcessed++; CDebugPrint( DebugLevelInfo, (CODECNAME ": Propagating data discontinuity, instance %u\n", i )); pOutData[0] = 0; pOutData[1] = 0; pOutStreamHeader->DataUsed = 2; pOutStreamHeader->OptionsFlags = pInStreamHeader->OptionsFlags | KSSTREAM_HEADER_OPTIONSF_DATADISCONTINUITY; CDebugPrint( DebugLevelInfo, (CODECNAME ": OptionsFlags %x\n", pOutStreamHeader->OptionsFlags )); CDebugPrint( DebugLevelInfo, ("" "Time %x Num %x Denom %x\n", pInStreamHeader->PresentationTime.Time, pInStreamHeader->PresentationTime.Numerator, pInStreamHeader->PresentationTime.Denominator )); RtlCopyMemory( &pOutStreamHeader->PresentationTime, &pInStreamHeader->PresentationTime, sizeof( pOutStreamHeader->PresentationTime )); pOutStreamHeader->Duration = pInStreamHeader->Duration; } CDebugPrint( DebugLevelVerbose, ( CODECNAME ": Releasing Output SRB %x\n", pOutSrb )); // Complete the output SRB
StreamClassStreamNotification( StreamRequestComplete, pOutSrb->StreamObject, pOutSrb ); pOutStrmEx->fDiscontinuity = FALSE; PinStats->Common.BytesOutput += pOutStreamHeader->DataUsed; Stats->Common.BytesOutput += pOutStreamHeader->DataUsed; } else { pOutStrmEx->fDiscontinuity = TRUE; Stats->Common.OutputSRBsMissing++; } }
return 1; }
return 0; }
void DebugPrintSubStreamMode( DWORD dwMode ) { #ifdef DEBUG
if ( dwMode & KS_CC_SUBSTREAM_SERVICE_XDS ) { CDebugPrint( DebugLevelWarning, ( "\n[XDS]" )); } if ( dwMode & KS_CC_SUBSTREAM_SERVICE_CC1 ) { CDebugPrint( DebugLevelWarning, ( "\n[CC1]" )); } if ( dwMode & KS_CC_SUBSTREAM_SERVICE_CC2 ) { CDebugPrint( DebugLevelWarning, ( "\n[CC2]" )); } if ( dwMode & KS_CC_SUBSTREAM_SERVICE_CC3 ) { CDebugPrint( DebugLevelWarning, ( "\n[CC3]" )); } if ( dwMode & KS_CC_SUBSTREAM_SERVICE_CC4 ) { CDebugPrint( DebugLevelWarning, ( "\n[CC4]" )); } if ( dwMode & KS_CC_SUBSTREAM_SERVICE_T1 ) { CDebugPrint( DebugLevelWarning, ( "\n[T1]" )); } if ( dwMode & KS_CC_SUBSTREAM_SERVICE_T2 ) { CDebugPrint( DebugLevelWarning, ( "\n[T2]" )); } if ( dwMode & KS_CC_SUBSTREAM_SERVICE_T3 ) { CDebugPrint( DebugLevelWarning, ( "\n[T3]" )); } if ( dwMode & KS_CC_SUBSTREAM_SERVICE_T4 ) { CDebugPrint( DebugLevelWarning, ( "\n[T4]" )); } #endif
}
// Get the substream mode for the current data sample - optionally change the substream mode
// The following "protocol" is loosely defined by FCC 91-119, FCC 92-157 and EIA 608
// Be advised that EIA 608 describes clearly which (few) byte pairs change the substream mode
DWORD GetSubStreamMode( DWORD dwFrameFlags, LPDWORD pdwCurrentSubStreamMode, PDSPRESULT pDSPResult ) { DWORD dwSubStreamMode = *pdwCurrentSubStreamMode; DWORD dwDataChannel = 0;
dwFrameFlags &= (KS_VBI_FLAG_FIELD1 | KS_VBI_FLAG_FIELD2);
if ( pDSPResult->Confidence >= 75 ) { // Inspect the first byte(minus parity) to see what substream this might be
switch ( pDSPResult->Decoded[0] & 0x7F ) { case CC_XDS_START_CURRENT: case CC_XDS_CONTINUE_CURRENT: case CC_XDS_START_FUTURE: case CC_XDS_CONTINUE_FUTURE: case CC_XDS_START_CHANNEL: case CC_XDS_CONTINUE_CHANNEL: case CC_XDS_START_MISC: case CC_XDS_CONTINUE_MISC: case CC_XDS_START_PUBLIC_SERVICE: case CC_XDS_CONTINUE_PUBLIC_SERVICE: case CC_XDS_START_RESERVED: case CC_XDS_CONTINUE_RESERVED: case CC_XDS_START_UNDEFINED: case CC_XDS_CONTINUE_UNDEFINED: case CC_XDS_END: // Set the substream mode as XDS from here on out
dwSubStreamMode = ( dwFrameFlags | KS_CC_SUBSTREAM_SERVICE_XDS); DebugPrintSubStreamMode( dwSubStreamMode ); *pdwCurrentSubStreamMode = dwSubStreamMode; break; case CC_MCC_FIELD1_DC1: dwDataChannel = KS_CC_SUBSTREAM_SERVICE_CC1; break; case CC_MCC_FIELD1_DC2: dwDataChannel = KS_CC_SUBSTREAM_SERVICE_CC2; break; case CC_MCC_FIELD2_DC1: dwDataChannel = KS_CC_SUBSTREAM_SERVICE_CC3; break; case CC_MCC_FIELD2_DC2: dwDataChannel = KS_CC_SUBSTREAM_SERVICE_CC4; break; }
// If we found a data channel escape, inspect the second byte(minus parity) to see what substream may be.
if ( dwDataChannel ) { switch ( pDSPResult->Decoded[1] & 0x7F ) { case CC_MCC_RCL: case CC_MCC_RU2: case CC_MCC_RU3: case CC_MCC_RU4: case CC_MCC_RDC: case CC_MCC_EOC: // The mode is good for this data pair and thereafter
dwSubStreamMode = (dwFrameFlags | dwDataChannel); DebugPrintSubStreamMode( dwSubStreamMode ); *pdwCurrentSubStreamMode = dwSubStreamMode; break; case CC_MCC_TR: case CC_MCC_RTD: // The mode is TEXT rather CC, map to the text channel ids
switch ( dwDataChannel ) { case KS_CC_SUBSTREAM_SERVICE_CC1: dwDataChannel = KS_CC_SUBSTREAM_SERVICE_T1; break; case KS_CC_SUBSTREAM_SERVICE_CC2: dwDataChannel = KS_CC_SUBSTREAM_SERVICE_T2; break; case KS_CC_SUBSTREAM_SERVICE_CC3: dwDataChannel = KS_CC_SUBSTREAM_SERVICE_T3; break; case KS_CC_SUBSTREAM_SERVICE_CC4: dwDataChannel = KS_CC_SUBSTREAM_SERVICE_T4; break; } // The mode is good for this data byte pair and thereafter
dwSubStreamMode = (dwFrameFlags | dwDataChannel); DebugPrintSubStreamMode( dwSubStreamMode ); *pdwCurrentSubStreamMode = dwSubStreamMode; break; case CC_MCC_EDM: case CC_MCC_ENM: // The mode is only good for this data byte pair. Reverts back thereafter
dwSubStreamMode = (dwFrameFlags | dwDataChannel); DebugPrintSubStreamMode( dwSubStreamMode ); DebugPrintSubStreamMode( *pdwCurrentSubStreamMode ); break; } } } return dwSubStreamMode; }
/*
** OutputCC ** ** Outputs just-decoded/received CC to any interested pins ** ** Arguments: ** ** PHW_DEVICE_EXTENSION pHwDevExt ** PSTREAMEX pInStrmEx ** PKS_VBI_FRAME_INFO pInVBIFrameInfo ** PKSSTREAM_HEADER pInStreamHeader ** ** Returns: nothing ** ** Side Effects: none */
void OutputCC( PHW_DEVICE_EXTENSION pHwDevExt, PSTREAMEX pInStrmEx, DWORD dwOriginalFrameFlags, PKSSTREAM_HEADER pInStreamHeader ) { ULONG i, ScanlineCount, CurrentStrmEx = 0; PKS_VBIINFOHEADER pVBIInfoHeader = &pInStrmEx->CurrentVBIInfoHeader; PVBICODECFILTERING_STATISTICS_CC Stats = 0;
#ifdef PERFTEST
ULONGLONG PerfStartTime = 0, PerfPreDownstreamCompletion = 0, PerfPostDownstreamCompletion = 0; LARGE_INTEGER PerfFrequency;
PerfStartTime = KeQueryPerformanceCounter( &PerfFrequency ).QuadPart; OldLevel = _CDebugLevel; _CDebugLevel = DebugLevelFatal; #endif // PERFTEST
CASSERT(pHwDevExt); CASSERT(pInStrmEx); Stats = &pHwDevExt->Statistics; CDebugPrint( DebugLevelInfo, ( "*" )); CDebugPrint( DebugLevelTrace, ( CODECNAME ": --->OutputCC\n" ));
// If this substream is requested by anybody (field or cc data channel)
if(( pInStrmEx->SubstreamsRequested.SubstreamMask )) { // Loop through all pending outbound requests and fill each irp with the requested data then complete the IO
for( ScanlineCount = pVBIInfoHeader->StartLine; ScanlineCount <= pVBIInfoHeader->EndLine; ScanlineCount++ ) { DWORD dwSubStreams = 0; DWORD dwFieldIndex = dwOriginalFrameFlags & KS_VBI_FLAG_FIELD1 ? 0 : 1; DWORD dwScanLineIndex = ScanlineCount - pHwDevExt->DSPResultStartLine;
if( !TESTBIT( pInStrmEx->ScanlinesRequested.DwordBitArray, ScanlineCount )) continue;
dwSubStreams = GetSubStreamMode( dwOriginalFrameFlags, &pHwDevExt->SubStreamState[dwScanLineIndex][dwFieldIndex], &pHwDevExt->DSPResult[dwScanLineIndex] );
CDebugPrint( DebugLevelWarning, ( "%c%c", pHwDevExt->DSPResult[dwScanLineIndex].Decoded[0] & 0x7f, pHwDevExt->DSPResult[dwScanLineIndex].Decoded[1] & 0x7f ));
CDebugPrint( DebugLevelInfo, (CODECNAME ": F%u %luus L%u %u%% %02x %02x\n", dwSubStreams & pInStrmEx->SubstreamsRequested.SubstreamMask, pVBIInfoHeader->ActualLineStartTime, ScanlineCount, pHwDevExt->DSPResult[dwScanLineIndex].Confidence, pHwDevExt->DSPResult[dwScanLineIndex].Decoded[0] & 0xff, pHwDevExt->DSPResult[dwScanLineIndex].Decoded[1] & 0xff ));
CurrentStrmEx = 0; for( i = 0; i < pHwDevExt->ActualInstances[STREAM_CC]; i++ ) { PSTREAMEX pOutStrmEx; PHW_STREAM_REQUEST_BLOCK pOutSrb; PVBICODECFILTERING_STATISTICS_CC_PIN PinStats;
do { CASSERT( CurrentStrmEx < MAX_PIN_INSTANCES ); if( CurrentStrmEx == MAX_PIN_INSTANCES ) Stats->Common.InternalErrors++; pOutStrmEx = pHwDevExt->pStrmEx[STREAM_CC][CurrentStrmEx++]; }while( !pOutStrmEx );
if( !TESTBIT( pOutStrmEx->ScanlinesRequested.DwordBitArray, ScanlineCount ) || !( pOutStrmEx->SubstreamsRequested.SubstreamMask & dwSubStreams )) continue;
PinStats = &pOutStrmEx->PinStats; //
// Update the average confidence for this pin
//
PinStats->Common.LineConfidenceAvg = ( PinStats->Common.LineConfidenceAvg + pHwDevExt->DSPResult[dwScanLineIndex].Confidence ) / 2; if( pHwDevExt->DSPResult[dwScanLineIndex].Confidence >= 75 ) { SETBIT( pInStrmEx->ScanlinesDiscovered.DwordBitArray, ScanlineCount ); SETBIT( pOutStrmEx->ScanlinesDiscovered.DwordBitArray, ScanlineCount ); SETBIT( pHwDevExt->ScanlinesDiscovered.DwordBitArray, ScanlineCount );
pInStrmEx->SubstreamsDiscovered.SubstreamMask |= dwSubStreams; pOutStrmEx->SubstreamsDiscovered.SubstreamMask |= dwSubStreams; pHwDevExt->SubstreamsDiscovered.SubstreamMask |= dwSubStreams; } else { Stats->Common.DSPFailures++; PinStats->Common.SRBsIgnored++; if(( dwSubStreams & KS_CC_SUBSTREAM_ODD ) && TESTBIT( pInStrmEx->LastOddScanlinesDiscovered.DwordBitArray, ScanlineCount )) pOutStrmEx->fDiscontinuity = TRUE; if(( dwSubStreams & KS_CC_SUBSTREAM_EVEN ) && TESTBIT( pInStrmEx->LastEvenScanlinesDiscovered.DwordBitArray, ScanlineCount )) pOutStrmEx->fDiscontinuity = TRUE; if( !pOutStrmEx->fDiscontinuity ) continue; }
// Only process the output streams which have an SRB ready
if( QueueRemove( &pOutSrb, &pOutStrmEx->StreamDataSpinLock, &pOutStrmEx->StreamDataQueue )) { PKSSTREAM_HEADER pOutStreamHeader = pOutSrb->CommandData.DataBufferArray; PKS_VBI_FRAME_INFO pOutVBIFrameInfo = (PKS_VBI_FRAME_INFO)(pOutStreamHeader+1); PUCHAR pOutData = (PUCHAR)pOutStreamHeader->Data;
PinStats->Common.SRBsProcessed++; Stats->Common.OutputSRBsProcessed++;
if( pOutStreamHeader->FrameExtent < pOutStrmEx->MatchedFormat.SampleSize ) { CDebugPrint( DebugLevelError, ( CODECNAME ": Outgoing Data SRB buffer is too small %u\n", pOutStreamHeader->FrameExtent )); PinStats->Common.InternalErrors++; Stats->Common.OutputFailures++; pOutStreamHeader->DataUsed = 0; } // Check on inbound & outbound data formats to decide
// whether to copy or decode the inbound data
// Figure out how much of the decoded data was requested
pOutStreamHeader->Size = pInStreamHeader->Size; pOutStreamHeader->OptionsFlags = pInStreamHeader->OptionsFlags; pOutStreamHeader->Duration = pInStreamHeader->Duration; RtlCopyMemory( &pOutStreamHeader->PresentationTime, &pInStreamHeader->PresentationTime, sizeof( pOutStreamHeader->PresentationTime )); // pOutData is the output location.
ASSERT( pOutStreamHeader->FrameExtent >= CCSamples ); pOutStreamHeader->DataUsed = 2; //
// If we have a discontinity to go out then send it
// instead of the data
//
if( pOutStrmEx->fDiscontinuity ) { PinStats->Common.Discontinuities++; pOutData[0] = 0xff; pOutData[1] = 0xff; pOutStreamHeader->OptionsFlags |= KSSTREAM_HEADER_OPTIONSF_DATADISCONTINUITY; pOutStrmEx->fDiscontinuity = FALSE; } else { pOutData[0] = pHwDevExt->DSPResult[ScanlineCount - pHwDevExt->DSPResultStartLine].Decoded[0]; pOutData[1] = pHwDevExt->DSPResult[ScanlineCount - pHwDevExt->DSPResultStartLine].Decoded[1]; } CDebugPrint( DebugLevelInfo, (CODECNAME ": OptionsFlags %x\n", pOutStreamHeader->OptionsFlags )); CDebugPrint( DebugLevelInfo, ("" "Time %x Num %x Denom %x\n", pInStreamHeader->PresentationTime.Time, pInStreamHeader->PresentationTime.Numerator, pInStreamHeader->PresentationTime.Denominator )); //CDebugPrint( DebugLevelWarning, ( "" "%d%", i ));
Stats->Common.BytesOutput += pOutStreamHeader->DataUsed; PinStats->Common.BytesOutput += pOutStreamHeader->DataUsed;
CDebugPrint( DebugLevelVerbose, ( CODECNAME ": Releasing Output SRB %x\n", pOutSrb )); // Complete the output SRB
#ifdef PERFTEST
if( i == 0 ) PerfPreDownstreamCompletion = KeQueryPerformanceCounter( NULL ).QuadPart; #endif // PERFTEST
StreamClassStreamNotification( StreamRequestComplete, pOutSrb->StreamObject, pOutSrb ); #ifdef PERFTEST
if( i == 0 ) PerfPostDownstreamCompletion = KeQueryPerformanceCounter( NULL ).QuadPart; #endif // PERFTEST
} else { PinStats->Common.SRBsMissing++; Stats->Common.OutputSRBsMissing++; } } } }
//
// Remember the streams we discovered so that if they aren't discovered next
// time we know we have to send a single discontinuity.
//
if(( dwOriginalFrameFlags & KS_CC_SUBSTREAM_EVEN ) == KS_CC_SUBSTREAM_EVEN ) RtlCopyMemory( &pInStrmEx->LastEvenScanlinesDiscovered, &pInStrmEx->ScanlinesDiscovered, sizeof( pInStrmEx->LastEvenScanlinesDiscovered )); if(( dwOriginalFrameFlags & KS_CC_SUBSTREAM_ODD ) == KS_CC_SUBSTREAM_ODD ) RtlCopyMemory( &pInStrmEx->LastOddScanlinesDiscovered, &pInStrmEx->ScanlinesDiscovered, sizeof( pInStrmEx->LastOddScanlinesDiscovered ));
#ifdef PERFTEST
PerfFrequency.QuadPart /= 1000000L; // Convert to ticks/us
if( PerfPreDownstreamCompletion ) { PerfPreDownstreamCompletion -= PerfStartTime; PerfPreDownstreamCompletion /= PerfFrequency.QuadPart; } if( PerfPostDownstreamCompletion ) { PerfPostDownstreamCompletion -= PerfStartTime; PerfPostDownstreamCompletion /= PerfFrequency.QuadPart; }
//
// Complain if anything takes more than threshold
//
if( PerfPreDownstreamCompletion > PerfThreshold ) CDebugPrint( DebugLevelFatal, ( CODECNAME ": PerfPreDownstreamCompletion %luus\n", PerfPreDownstreamCompletion )); if( PerfPostDownstreamCompletion > PerfThreshold ) CDebugPrint( DebugLevelFatal, ( CODECNAME ": PerfPostDownstreamCompletion %luus\n", PerfPostDownstreamCompletion )); _CDebugLevel = OldLevel; #endif // PERFTEST
CDebugPrint( DebugLevelTrace, ( CODECNAME ": <---OutputCC\n" )); }
/*
** VBIDecode ** ** Decodes an incoming SRB. SRB is already removed from queue. ** ** Arguments: ** ** PHW_DEVICE_EXTENSION pHwDevExt ** PSTREAMEX pInStrmEx ** IN PHW_STREAM_REQUEST_BLOCK pInSrb ** ** Returns: ** ** Side Effects: none */
#ifdef DEBUG
short CCskipDecode = 0; #endif /*DEBUG*/
void VBIDecode( PHW_DEVICE_EXTENSION pHwDevExt, PSTREAMEX pInStrmEx, PHW_STREAM_REQUEST_BLOCK pInSrb, BOOL OkToHold ) { PKSSTREAM_HEADER pInStreamHeader = pInSrb->CommandData.DataBufferArray; KSSTREAM_HEADER InStreamHeaderCopy; PKS_VBI_FRAME_INFO pInVBIFrameInfo = (PKS_VBI_FRAME_INFO)(pInStreamHeader+1); DWORD dwFrameFlags; PUCHAR pInData = (PUCHAR)pInStreamHeader->Data; ULONG i, j, ScanlineCount, DSPStatus, CurrentStrmEx = 0; CCLineStats DSPStatistics; PKS_VBIINFOHEADER pVBIInfoHeader = &pInStrmEx->CurrentVBIInfoHeader; PVBICODECFILTERING_STATISTICS_CC Stats = 0;
CASSERT(KeGetCurrentIrql() <= APC_LEVEL);
#ifdef PERFTEST
ULONGLONG PerfStartTime = 0, PerfPreUpstreamCompletion = 0, PerfPostUpstreamCompletion = 0; LARGE_INTEGER PerfFrequency;
PerfStartTime = KeQueryPerformanceCounter( &PerfFrequency ).QuadPart; OldLevel = _CDebugLevel; _CDebugLevel = DebugLevelFatal; #endif // PERFTEST
CASSERT(pHwDevExt); CASSERT(pInStrmEx); Stats = &pHwDevExt->Statistics;
CDebugPrint( DebugLevelTrace, ( CODECNAME ": --->VBIDecode\n" )); #ifdef CCINPUTPIN
if (!OkToHold) goto GoodToGo; // We've already processed discontinuties & stuff below
#endif // CCINPUTPIN
CDebugPrint( DebugLevelInfo, ( "*" ));
Stats->Common.InputSRBsProcessed++;
//
// If DataUsed == 0 then don't bother
//
if( pInStreamHeader->DataUsed < 1 #ifdef DEBUG
|| CCskipDecode #endif /*DEBUG*/
) { Stats->Common.SRBsIgnored++; #ifdef DEBUG
if (!CCskipDecode) #endif /*DEBUG*/
CDebugPrint( DebugLevelError, ( CODECNAME ": DataUsed == 0, abandoning\n" )); StreamClassStreamNotification( StreamRequestComplete, pInSrb->StreamObject, pInSrb ); return; }
//
// Test for dropped fields
//
if( pInStrmEx->LastPictureNumber ) { LONGLONG Dropped = pInVBIFrameInfo->PictureNumber - pInStrmEx->LastPictureNumber - 1; if( Dropped > 0 ) { if( Dropped < 60*60*60 ) // One hour worth of video fields
Stats->Common.InputSRBsMissing += (DWORD)Dropped; else Stats->Common.InputSRBsMissing++; // Some improbable number of fields got dropped, indicate a single lost field.
CDebugPrint( DebugLevelWarning, ( "$" )); } } pInStrmEx->LastPictureNumber = pInVBIFrameInfo->PictureNumber;
CDebugPrint( DebugLevelVerbose, ( CODECNAME ": pInVBIFrameInfo->ExtendedHeaderSize %d\n", pInVBIFrameInfo->ExtendedHeaderSize )); CDebugPrint( DebugLevelVerbose, ( CODECNAME ": pInVBIFrameInfo->dwFrameFlags %x\n", pInVBIFrameInfo->dwFrameFlags )); CDebugPrint( DebugLevelVerbose, ( CODECNAME ": pInVBIFrameInfo->PictureNumber %lu\n", pInVBIFrameInfo->PictureNumber )); CDebugPrint( DebugLevelVerbose, ( CODECNAME ": pInVBIFrameInfo->DropCount %lu\n", pInVBIFrameInfo->DropCount )); CDebugPrint( DebugLevelVerbose, ( CODECNAME ": pInVBIFrameInfo->dwSamplingFrequency %lu\n", pInVBIFrameInfo->dwSamplingFrequency ));
CDebugPrint( DebugLevelVerbose, ( CODECNAME ": pInStreamHeader->FrameExtent %d\n", pInStreamHeader->FrameExtent ));
//
// Update stats
//
if( ( pInStreamHeader->OptionsFlags & KSSTREAM_HEADER_OPTIONSF_DATADISCONTINUITY ) || ( pInStreamHeader->OptionsFlags & KSSTREAM_HEADER_OPTIONSF_TIMEDISCONTINUITY ) ) Stats->Common.InputDiscontinuities++;
//
// Check for a new VBIINFOHEADER
//
if( pInVBIFrameInfo->dwFrameFlags & KS_VBI_FLAG_VBIINFOHEADER_CHANGE ) { CheckNewVBIInfo( pHwDevExt, pInStrmEx, pInVBIFrameInfo ); }
//
// Check for a channel change
//
if( pInVBIFrameInfo->dwFrameFlags & KS_VBI_FLAG_TVTUNER_CHANGE ) { if( ProcessChannelChange( pHwDevExt, pInStrmEx, pInVBIFrameInfo, pInStreamHeader )) { StreamClassStreamNotification( StreamRequestComplete, pInSrb->StreamObject, pInSrb ); return; } }
//
// pHwDevExt->fTunerChange is set while the TV tuner is changing channels.
// SRBs are junk until the channel change completes so complete.
//
if( pHwDevExt->fTunerChange) { CDebugPrint( DebugLevelVerbose, ( CODECNAME ": Completing, channel change in progress\n" ));
StreamClassStreamNotification( StreamRequestComplete, pInSrb->StreamObject, pInSrb ); return; }
// Clear the current scanline & substream request masks
RtlZeroMemory( &pInStrmEx->ScanlinesRequested, sizeof(pInStrmEx->ScanlinesRequested) ); RtlZeroMemory( &pInStrmEx->SubstreamsRequested, sizeof(pInStrmEx->SubstreamsRequested) );
//
// Calculate the current request from union of the output pins w/pending SRBs that are
// interested in this substream.
//
CurrentStrmEx = 0; for( i = 0; i < pHwDevExt->ActualInstances[STREAM_CC]; i++ ) { PSTREAMEX pOutStrmEx;
do { CASSERT( CurrentStrmEx < MAX_PIN_INSTANCES ); pOutStrmEx = pHwDevExt->pStrmEx[STREAM_CC][CurrentStrmEx++]; }while( !pOutStrmEx );
if ( pInVBIFrameInfo->dwFrameFlags & KS_VBI_FLAG_TVTUNER_CHANGE ) pOutStrmEx->fDiscontinuity = TRUE; //
// For actual processing, just include the scanlines of the clients who are
// interested in this particular substream.
//
if ( ( ( pInVBIFrameInfo->dwFrameFlags & KS_CC_SUBSTREAM_ODD ) && ( pOutStrmEx->SubstreamsRequested.SubstreamMask & (KS_CC_SUBSTREAM_ODD|KS_CC_SUBSTREAM_FIELD1_MASK) ) ) || ( ( pInVBIFrameInfo->dwFrameFlags & KS_CC_SUBSTREAM_EVEN ) && ( pOutStrmEx->SubstreamsRequested.SubstreamMask & (KS_CC_SUBSTREAM_EVEN|KS_CC_SUBSTREAM_FIELD2_MASK) ) ) ) { for( j = 0; j < SIZEOF_ARRAY( pInStrmEx->ScanlinesRequested.DwordBitArray ); j++ ) pInStrmEx->ScanlinesRequested.DwordBitArray[j] |= pOutStrmEx->ScanlinesRequested.DwordBitArray[j];
// Create the union of all the requested substreams
pInStrmEx->SubstreamsRequested.SubstreamMask |= pOutStrmEx->SubstreamsRequested.SubstreamMask; } } // Decode the union of all the pending decode requests into a local decode buffer.
#ifdef CCINPUTPIN
GoodToGo: // Whoever gets there first (VBI pin vs. HW pin) supplies CC data
ExAcquireFastMutex(&pHwDevExt->LastPictureMutex); if (pInStrmEx->LastPictureNumber <= pHwDevExt->LastPictureNumber) { // HW pin beat us to it
ExReleaseFastMutex(&pHwDevExt->LastPictureMutex); StreamClassStreamNotification( StreamRequestComplete, pInSrb->StreamObject, pInSrb ); return; }
// Is the HW stream open?
if (OkToHold && pHwDevExt->ActualInstances[STREAM_CCINPUT] > 0) { KIRQL Irql;
// We're going to give the HW pin a chance to catch up
ExReleaseFastMutex(&pHwDevExt->LastPictureMutex);
KeAcquireSpinLock(&pInStrmEx->VBIOnHoldSpinLock, &Irql); CASSERT(NULL == pInStrmEx->pVBISrbOnHold); pInStrmEx->pVBISrbOnHold = pInSrb; KeReleaseSpinLock(&pInStrmEx->VBIOnHoldSpinLock, Irql);
return; }
// HW input pin not open or too late; we'll process this SRB
pHwDevExt->LastPictureNumber = pInStrmEx->LastPictureNumber; ExReleaseFastMutex(&pHwDevExt->LastPictureMutex); #endif // CCINPUTPIN
CDebugPrint( DebugLevelTrace, ( CODECNAME ": Requested SubstreamMask %x\n", pInStrmEx->SubstreamsRequested.SubstreamMask )); CDebugPrint( DebugLevelTrace, ( CODECNAME ": Requested Scanlines %08x%08x\n", pInStrmEx->ScanlinesRequested.DwordBitArray[1], pInStrmEx->ScanlinesRequested.DwordBitArray[0] ));
RtlZeroMemory( pHwDevExt->DSPResult, sizeof( DSPRESULT ) * ( pHwDevExt->DSPResultEndLine - pHwDevExt->DSPResultStartLine + 1 ));
// If this substream is requested by anybody, AND there is no discontinuity set
if ( ( ( ( pInVBIFrameInfo->dwFrameFlags & KS_CC_SUBSTREAM_ODD ) && ( pInStrmEx->SubstreamsRequested.SubstreamMask & (KS_CC_SUBSTREAM_ODD|KS_CC_SUBSTREAM_FIELD1_MASK) ) || ( ( pInVBIFrameInfo->dwFrameFlags & KS_CC_SUBSTREAM_EVEN ) && ( pInStrmEx->SubstreamsRequested.SubstreamMask & (KS_CC_SUBSTREAM_EVEN|KS_CC_SUBSTREAM_FIELD2_MASK) ) ) ) && !pInStrmEx->fDiscontinuity )) { // Flag this as discovered
// pInStrmEx->SubstreamsDiscovered.SubstreamMask |= ( pInVBIFrameInfo->dwFrameFlags &
// pInStrmEx->SubstreamsRequested.SubstreamMask );
// pHwDevExt->SubstreamsDiscovered.SubstreamMask |= pInStrmEx->SubstreamsDiscovered.SubstreamMask;
// loop for each requested scanline
CDebugPrint( DebugLevelVerbose, ( "" "\n" )); for( ScanlineCount = pVBIInfoHeader->StartLine; ScanlineCount <= pVBIInfoHeader->EndLine; ScanlineCount++ ) { if( !TESTBIT( pInStrmEx->ScanlinesRequested.DwordBitArray, ScanlineCount )) continue; CDebugPrint( DebugLevelTrace, ( CODECNAME ": Scanning %u\n", ScanlineCount )); CASSERT( ( ScanlineCount - pVBIInfoHeader->StartLine) * pVBIInfoHeader->StrideInBytes < pVBIInfoHeader->BufferSize ); DSPStatistics.nSize = sizeof( DSPStatistics ); DSPStatus = CCDecodeLine( pHwDevExt->DSPResult[ScanlineCount - pHwDevExt->DSPResultStartLine].Decoded, &DSPStatistics, &pInData[( ScanlineCount - pVBIInfoHeader->StartLine ) * pVBIInfoHeader->StrideInBytes], &pInStrmEx->State, pVBIInfoHeader ); CASSERT( DSPStatus == CC_OK ); if( DSPStatus == CC_OK ) { pHwDevExt->DSPResult[ScanlineCount - pHwDevExt->DSPResultStartLine].Confidence = DSPStatistics.nConfidence; Stats->Common.LineConfidenceAvg = ( Stats->Common.LineConfidenceAvg + DSPStatistics.nConfidence ) / 2; } else Stats->Common.InternalErrors++; } } else Stats->Common.SRBsIgnored++;
//
// Copy the input stream header info for later reference
//
InStreamHeaderCopy = *pInStreamHeader; dwFrameFlags = pInVBIFrameInfo->dwFrameFlags;
#ifdef PERFTEST
PerfPreUpstreamCompletion = KeQueryPerformanceCounter( NULL ).QuadPart; #endif // PERFTEST
//
// Complete the upstream SRB.
//
StreamClassStreamNotification( StreamRequestComplete, pInSrb->StreamObject, pInSrb );
#ifdef PERFTEST
PerfPostUpstreamCompletion = KeQueryPerformanceCounter( NULL ).QuadPart; #endif // PERFTEST
//
// Lose all references to the just completed SRB
//
pInSrb = 0; pInStreamHeader = 0; pInVBIFrameInfo = 0; pInData = 0;
#ifdef PERFTEST
PerfFrequency.QuadPart /= 1000000L; // Convert to ticks/us
PerfPreUpstreamCompletion -= PerfStartTime; PerfPreUpstreamCompletion /= PerfFrequency.QuadPart; PerfPostUpstreamCompletion -= PerfStartTime; PerfPostUpstreamCompletion /= PerfFrequency.QuadPart;
//
// Complain if anything takes more than threshold
//
if( PerfPreUpstreamCompletion > PerfThreshold ) CDebugPrint( DebugLevelFatal, ( CODECNAME ": PerfPreUpstreamCompletion %luus\n", PerfPreUpstreamCompletion )); if( PerfPostUpstreamCompletion > PerfThreshold ) CDebugPrint( DebugLevelFatal, ( CODECNAME ": PerfPostUpstreamCompletion %luus\n", PerfPostUpstreamCompletion )); _CDebugLevel = OldLevel; #endif // PERFTEST
//
// Now output to anyone interested
//
OutputCC(pHwDevExt, pInStrmEx, dwFrameFlags, &InStreamHeaderCopy);
CDebugPrint( DebugLevelTrace, ( CODECNAME ": <---VBIDecode\n" )); }
/*
** VBIhwDecode ** ** Handles an incoming CCINPUT SRB. SRB is already removed from queue. ** ** Arguments: ** ** PHW_DEVICE_EXTENSION pHwDevExt ** PSTREAMEX pInStrmEx ** IN PHW_STREAM_REQUEST_BLOCK pInSrb ** ** Returns: ** ** Side Effects: none */
#ifdef CCINPUTPIN
#ifdef DEBUG
short CCskipHwDecode = 0; #endif /*DEBUG*/
#ifdef NEWCCINPUTFORMAT
void VBIhwDecode( PHW_DEVICE_EXTENSION pHwDevExt, PSTREAMEX pInStrmEx, PHW_STREAM_REQUEST_BLOCK pInSrb ) { PKSSTREAM_HEADER pInStreamHeader = pInSrb->CommandData.DataBufferArray; KSSTREAM_HEADER InStreamHeaderCopy; PCC_HW_FIELD pCCin = (PCC_HW_FIELD)pInStreamHeader->Data; ULONG CurrentStrmEx = 0; PVBICODECFILTERING_STATISTICS_CC Stats = 0; int line, start, end; int hidx; int didx; DWORD fields;
CASSERT(KeGetCurrentIrql() <= APC_LEVEL);
CASSERT(pHwDevExt); CASSERT(pInStrmEx); Stats = &pHwDevExt->Statistics; CDebugPrint( DebugLevelInfo, ( "*" )); CDebugPrint( DebugLevelTrace, ( CODECNAME ": --->VBIhwDecode\n" ));
Stats->Common.InputSRBsProcessed++;
//
// If DataUsed == 0 then don't bother
//
if( pInStreamHeader->DataUsed < sizeof (CC_HW_FIELD) #ifdef DEBUG
|| CCskipHwDecode #endif /*DEBUG*/
) { #ifdef DEBUG
if (!CCskipHwDecode) #endif /*DEBUG*/
{ Stats->Common.SRBsIgnored++; CDebugPrint( DebugLevelError, ( CODECNAME ": DataUsed is too small, abandoning\n" )); } StreamClassStreamNotification( StreamRequestComplete, pInSrb->StreamObject, pInSrb ); return; } pInStrmEx->LastPictureNumber = pCCin->PictureNumber;
//
// Update stats
//
if( ( pInStreamHeader->OptionsFlags & KSSTREAM_HEADER_OPTIONSF_DATADISCONTINUITY ) || ( pInStreamHeader->OptionsFlags & KSSTREAM_HEADER_OPTIONSF_TIMEDISCONTINUITY ) ) Stats->Common.InputDiscontinuities++;
//
// pHwDevExt->fTunerChange is set while the TV tuner is changing channels.
// SRBs are junk until the channel change completes so complete.
//
if( pHwDevExt->fTunerChange ) { CDebugPrint( DebugLevelVerbose, ( CODECNAME ": Completing, channel change in progress\n" )); StreamClassStreamNotification( StreamRequestComplete, pInSrb->StreamObject, pInSrb ); return; }
#ifdef CCINPUTPIN
// Check to see if this field has been decoded already (are we too late?)
ExAcquireFastMutex(&pHwDevExt->LastPictureMutex); if (pInStrmEx->LastPictureNumber <= pHwDevExt->LastPictureNumber) { ExReleaseFastMutex(&pHwDevExt->LastPictureMutex); StreamClassStreamNotification( StreamRequestComplete, pInSrb->StreamObject, pInSrb ); return; }
// Nope, we're not too late. Stow the data.
pHwDevExt->LastPictureNumber = pInStrmEx->LastPictureNumber; ExReleaseFastMutex(&pHwDevExt->LastPictureMutex); #endif // CCINPUTPIN
// figure out where hardware decoding starts and ends
for( start = 1; start < 1024; ++start ) { if( TESTBIT( pCCin->ScanlinesRequested.DwordBitArray, start )) break; } for( end = 1023; end > start; --end ) { if( TESTBIT( pCCin->ScanlinesRequested.DwordBitArray, end )) break; } if (1024 == start) { StreamClassStreamNotification( StreamRequestComplete, pInSrb->StreamObject, pInSrb ); return; } CASSERT(start <= end);
// Resize Result array if needed
CheckResultsArray(pHwDevExt, start, end);
// loop for each scanline
CDebugPrint( DebugLevelVerbose, ( "" "\n" ));
hidx = 0; for( line = start; line <= end && hidx < CC_MAX_HW_DECODE_LINES; ++line ) { if( !TESTBIT( pCCin->ScanlinesRequested.DwordBitArray, line )) continue; CDebugPrint( DebugLevelTrace, ( CODECNAME ": Scanning %u\n", line ));
didx = line - pHwDevExt->DSPResultStartLine; pHwDevExt->DSPResult[didx].Decoded[0] = pCCin->Lines[hidx].Decoded[0]; pHwDevExt->DSPResult[didx].Decoded[1] = pCCin->Lines[hidx].Decoded[1]; ++hidx;
pHwDevExt->DSPResult[didx].Confidence = 99; // HW decoded
Stats->Common.LineConfidenceAvg = (Stats->Common.LineConfidenceAvg + 99) / 2; }
//
// Copy the input stream header & other info for later reference
//
InStreamHeaderCopy = *pInStreamHeader; fields = pCCin->fieldFlags & (KS_VBI_FLAG_FIELD1|KS_VBI_FLAG_FIELD2);
//
// Complete the upstream SRB.
//
StreamClassStreamNotification( StreamRequestComplete, pInSrb->StreamObject, pInSrb );
//
// Lose all references to the just completed SRB
//
pInSrb = 0; pInStreamHeader = 0; pCCin = 0;
OutputCC(pHwDevExt, pInStrmEx, fields, &InStreamHeaderCopy);
CDebugPrint( DebugLevelTrace, ( CODECNAME ": <---VBIhwDecode\n" )); }
#else //NEWCCINPUTFORMAT
void VBIhwDecode( PHW_DEVICE_EXTENSION pHwDevExt, PSTREAMEX pInStrmEx, PHW_STREAM_REQUEST_BLOCK pInSrb ) { PKSSTREAM_HEADER pInStreamHeader = pInSrb->CommandData.DataBufferArray; KSSTREAM_HEADER InStreamHeaderCopy; PUCHAR pInData = (PUCHAR)pInStreamHeader->Data; ULONG CurrentStrmEx = 0; PVBICODECFILTERING_STATISTICS_CC Stats = 0;
CASSERT((ULONG)pHwDevExt); CASSERT((ULONG)pInStrmEx); Stats = &pHwDevExt->Statistics; CDebugPrint( DebugLevelInfo, ( "*" )); CDebugPrint( DebugLevelTrace, ( CODECNAME ": --->VBIhwDecode\n" ));
Stats->Common.InputSRBsProcessed++;
//
// If DataUsed == 0 then don't bother
//
if( pInStreamHeader->DataUsed < 2 #ifdef DEBUG
|| CCskipHwDecode #endif /*DEBUG*/
) { #ifdef DEBUG
if (!CCskipDecode) #endif /*DEBUG*/
Stats->Common.SRBsIgnored++; CDebugPrint( DebugLevelError, ( CODECNAME ": DataUsed is too small, abandoning\n" )); StreamClassStreamNotification( StreamRequestComplete, pInSrb->StreamObject, pInSrb ); return; }
//
// Update stats
//
if( ( pInStreamHeader->OptionsFlags & KSSTREAM_HEADER_OPTIONSF_DATADISCONTINUITY ) || ( pInStreamHeader->OptionsFlags & KSSTREAM_HEADER_OPTIONSF_TIMEDISCONTINUITY ) ) Stats->Common.InputDiscontinuities++;
//
// pHwDevExt->fTunerChange is set while the TV tuner is changing channels.
// SRBs are junk until the channel change completes so complete.
//
if( pHwDevExt->fTunerChange ) { CDebugPrint( DebugLevelVerbose, ( CODECNAME ": Completing, channel change in progress\n" )); StreamClassStreamNotification( StreamRequestComplete, pInSrb->StreamObject, pInSrb ); return; }
// loop for each requested scanline
CDebugPrint( DebugLevelVerbose, ( "" "\n" ));
pHwDevExt->DSPResult[21-10].Decoded[0] = pInData[0]; pHwDevExt->DSPResult[21-10].Decoded[1] = pInData[1]; pHwDevExt->DSPResult[21-10].Confidence = 95; Stats->Common.LineConfidenceAvg = ( Stats->Common.LineConfidenceAvg + pHwDevExt->DSPResult[21-10].Confidence ) / 2;
//
// Copy the input stream header for later reference
//
InStreamHeaderCopy = *pInStreamHeader;
//
// Complete the upstream SRB.
//
StreamClassStreamNotification( StreamRequestComplete, pInSrb->StreamObject, pInSrb );
//
// Lose all references to the just completed SRB
//
pInSrb = 0; pInStreamHeader = 0; pInData = 0;
//
// Now output to anyone interested
//
OutputCC(pHwDevExt, pInStrmEx, KS_VBI_FLAG_FIELD1, &InStreamHeaderCopy);
CDebugPrint( DebugLevelTrace, ( CODECNAME ": <---VBIhwDecode\n" )); }
#endif //NEWCCINPUTFORMAT
#endif //CCINPUTPIN
/*
** VBIReceiveDataPacket() ** ** Receives Video data packet commands ** ** Arguments: ** ** pSrb - Stream request block for the Video stream ** ** Returns: nothing ** ** Side Effects: none */
VOID STREAMAPI VBIReceiveDataPacket( IN PHW_STREAM_REQUEST_BLOCK pSrb ) { PHW_DEVICE_EXTENSION pHwDevExt = (PHW_DEVICE_EXTENSION)pSrb->HwDeviceExtension; PSTREAMEX pStrmEx = (PSTREAMEX)pSrb->StreamObject->HwStreamExtension; int ThisStreamNr = (int)pSrb->StreamObject->StreamNumber; #ifdef DEBUG
static int QdepthReportFreq = 0; static unsigned int QDRCount = 0; #endif // DEBUG
//
// make sure we have a device extension
//
CASSERT(pHwDevExt);
CDebugPrint(DebugLevelTrace,( CODECNAME ":--->VBIReceiveDataPacket(pSrb=%x)\n", pSrb));
//
// Default to success
//
pSrb->Status = STATUS_SUCCESS; //
// Disable timeout
//
pSrb->TimeoutCounter = 0;
//
// determine the type of packet.
//
// Rule:
// Only accept read requests when in either the Pause or Run
// States. If Stopped, immediately return the SRB.
if (pStrmEx->KSState == KSSTATE_STOP) { StreamClassStreamNotification( StreamRequestComplete, pSrb->StreamObject, pSrb );
return; }
switch (pSrb->Command) { #ifdef DRIVER_DEBUGGING_TEST
case SRB_READ_DATA: case SRB_WRITE_DATA: // When initially bringing up a driver, it is useful to just
// try immediately completing the SRB, thus verifying
// the streaming process independent of really accessing
// your hardware.
StreamClassStreamNotification( StreamRequestComplete, pSrb->StreamObject, pSrb );
break; #else // DRIVER_DEBUGGING_TEST
case SRB_READ_DATA: if( ThisStreamNr != STREAM_CC ) { CDebugPrint( DebugLevelError, ( CODECNAME ": Read Stream # Bad\n" )); CDEBUG_BREAK(); pSrb->Status = STATUS_NOT_IMPLEMENTED; } else { CDebugPrint( DebugLevelTrace, ( CODECNAME ": Stream %u Instance %u\n", ThisStreamNr, pStrmEx->StreamInstance )); if( pStrmEx->StreamInstance != 0 ) CDebugPrint( DebugLevelTrace, ( CODECNAME ": Stream %u Instance %u\n", ThisStreamNr, pStrmEx->StreamInstance )); QueueAdd( pSrb, &pStrmEx->StreamDataSpinLock, &pStrmEx->StreamDataQueue );
// Since another thread COULD HAVE MODIFIED THE STREAM STATE
// in the midst of adding it to the queue, check the stream
// state again, and cancel the SRB if necessary.
// Note that this race condition was NOT handled in the
// original DDK release of testcap!
if (pStrmEx->KSState == KSSTATE_STOP) CodecCancelPacket(pSrb); } break;
case SRB_WRITE_DATA: if( STREAM_VBI == ThisStreamNr) { #ifdef DEBUG
static int MaxVBIqDepth = 0; static int AvgVBIqDepth = 1000; // 1.000
int qDepth = 0; #endif // DEBUG
CDebugPrint( DebugLevelTrace, ( CODECNAME ": Stream VBI Writing\n")); if( QueueAddIfNotEmpty( pSrb, &pStrmEx->StreamDataSpinLock, &pStrmEx->StreamDataQueue )) break;
do { #ifdef CCINPUTPIN
KIRQL Irql; #endif // CCINPUTPIN
#ifdef DEBUG
++qDepth; ++QDRCount; #endif // DEBUG
#ifdef CCINPUTPIN
KeAcquireSpinLock(&pStrmEx->VBIOnHoldSpinLock, &Irql); if (NULL != pStrmEx->pVBISrbOnHold) { PHW_STREAM_REQUEST_BLOCK pTempSrb;
pTempSrb = pStrmEx->pVBISrbOnHold; pStrmEx->pVBISrbOnHold = NULL; KeReleaseSpinLock(&pStrmEx->VBIOnHoldSpinLock, Irql);
VBIDecode( pHwDevExt, pStrmEx, pTempSrb, 0 ); } else KeReleaseSpinLock(&pStrmEx->VBIOnHoldSpinLock, Irql); #endif // CCINPUTPIN
VBIDecode( pHwDevExt, pStrmEx, pSrb, 1 ); }while( QueueRemove( &pSrb, &pStrmEx->StreamDataSpinLock, &pStrmEx->StreamDataQueue )); #ifdef DEBUG
if (qDepth > MaxVBIqDepth) MaxVBIqDepth = qDepth; AvgVBIqDepth = (AvgVBIqDepth * 7 / 8) + (qDepth * 1000 / 8); if (QdepthReportFreq > 0 && 0 == QDRCount % QdepthReportFreq) { CDebugPrint( 0, (CODECNAME ": Max VBI Q depth = %3d, Avg VBI Q depth = %3d.%03d\n", MaxVBIqDepth, AvgVBIqDepth / 1000, AvgVBIqDepth % 1000)); } #endif // DEBUG
} #ifdef CCINPUTPIN
else if (STREAM_CCINPUT == ThisStreamNr) { #ifdef DEBUG
static int MaxCCINqDepth = 0; static int AvgCCINqDepth = 1000; // 1.000
int qDepth = 0; #endif // DEBUG
CDebugPrint( DebugLevelTrace, (CODECNAME ": Stream CCINPUT Writing\n")); if( QueueAddIfNotEmpty( pSrb, &pStrmEx->StreamDataSpinLock, &pStrmEx->StreamDataQueue )) break;
do { #ifdef DEBUG
++qDepth; ++QDRCount; #endif // DEBUG
VBIhwDecode( pHwDevExt, pStrmEx, pSrb ); }while( QueueRemove( &pSrb, &pStrmEx->StreamDataSpinLock, &pStrmEx->StreamDataQueue ));
#ifdef DEBUG
if (qDepth > MaxCCINqDepth) MaxCCINqDepth = qDepth; AvgCCINqDepth = (AvgCCINqDepth * 7 / 8) + (qDepth * 1000 / 8); if (QdepthReportFreq > 0 && 0 == QDRCount % QdepthReportFreq) { CDebugPrint( 0, (CODECNAME ": Max CCIN Q depth = %3d, Avg CCIN Q depth = %3d.%03d\n", MaxCCINqDepth, AvgCCINqDepth / 1000, AvgCCINqDepth % 1000)); } #endif // DEBUG
} #endif // CCINPUTPIN
else { CDebugPrint( DebugLevelError, ( CODECNAME, ": Write Stream # Bad (%u)\n", ThisStreamNr )); CDEBUG_BREAK(); pSrb->Status = STATUS_NOT_IMPLEMENTED; } break; #endif // DRIVER_DEBUGGING_TEST
break;
default:
//
// invalid / unsupported command. Fail it as such
//
CDEBUG_BREAK();
pSrb->Status = STATUS_NOT_IMPLEMENTED; StreamClassStreamNotification( StreamRequestComplete, pSrb->StreamObject, pSrb );
} // switch (pSrb->Command)
CDebugPrint(DebugLevelTrace,( CODECNAME ":<---VBIReceiveDataPacket(pSrb=%x)\n", pSrb)); }
/*
** VBIReceiveCtrlPacket() ** ** Receives packet commands that control the Video stream ** ** Arguments: ** ** pSrb - The stream request block for the Video stream ** ** Returns: nothing ** ** Side Effects: none */
VOID STREAMAPI VBIReceiveCtrlPacket( IN PHW_STREAM_REQUEST_BLOCK pSrb ) { PHW_DEVICE_EXTENSION pHwDevExt = ((PHW_DEVICE_EXTENSION)pSrb->HwDeviceExtension); PSTREAMEX pStrmEx = (PSTREAMEX)pSrb->StreamObject->HwStreamExtension;
CDebugPrint(DebugLevelTrace,( CODECNAME ":--->VBIReceiveCtrlPacket(pSrb=%x)\n", pSrb));
CASSERT(pHwDevExt);
//
// Default to success
//
pSrb->Status = STATUS_SUCCESS;
if( QueueAddIfNotEmpty( pSrb, &pStrmEx->StreamControlSpinLock, &pStrmEx->StreamControlQueue )) return;
do { //
// determine the type of packet.
//
switch (pSrb->Command) { case SRB_PROPOSE_DATA_FORMAT: if ( !CodecVerifyFormat( pSrb->CommandData.OpenFormat, pSrb->StreamObject->StreamNumber, NULL ) ) { pSrb->Status = STATUS_NO_MATCH; } break; case SRB_SET_STREAM_STATE:
VideoSetState(pSrb); break;
case SRB_GET_STREAM_STATE:
VideoGetState(pSrb); break;
case SRB_GET_STREAM_PROPERTY:
VideoGetProperty(pSrb); break;
case SRB_SET_STREAM_PROPERTY:
VideoSetProperty(pSrb); break;
case SRB_INDICATE_MASTER_CLOCK:
//
// Assigns a clock to a stream
//
VideoIndicateMasterClock(pSrb);
break;
default:
//
// invalid / unsupported command. Fail it as such
//
CDEBUG_BREAK();
pSrb->Status = STATUS_NOT_IMPLEMENTED; }
StreamClassStreamNotification( StreamRequestComplete, pSrb->StreamObject, pSrb ); }while( QueueRemove( &pSrb, &pStrmEx->StreamControlSpinLock, &pStrmEx->StreamControlQueue )); CDebugPrint(DebugLevelTrace,( CODECNAME ":<---VBIReceiveCtrlPacket(pSrb=%x)\n", pSrb)); }
/*
** VideoGetProperty() ** ** Routine to process video property requests ** ** Arguments: ** ** pSrb - pointer to the stream request block for properties ** ** Returns: ** ** Side Effects: none */
VOID VideoGetProperty( PHW_STREAM_REQUEST_BLOCK pSrb ) { PSTREAM_PROPERTY_DESCRIPTOR pSPD = pSrb->CommandData.PropertyInfo; int StreamNumber = (int)pSrb->StreamObject->StreamNumber;
CDebugPrint(DebugLevelTrace,( CODECNAME ":--->VideoGetProperty(pSrb=%x)\n", pSrb));
if (IsEqualGUID (&KSPROPSETID_Connection, &pSPD->Property->Set)) { VideoStreamGetConnectionProperty( pSrb ); } else if (IsEqualGUID (&KSPROPSETID_VBICodecFiltering, &pSPD->Property->Set)) { VideoStreamGetVBIFilteringProperty (pSrb); } else { CDebugPrint( DebugLevelTrace, ( CODECNAME ": Unsupported Property Set\n" )); pSrb->Status = STATUS_NOT_IMPLEMENTED; }
CDebugPrint(DebugLevelTrace,( CODECNAME ":<---VideoGetProperty(pSrb=%x)\n", pSrb)); }
/*
** VideoSetProperty() ** ** Routine to process video property requests ** ** Arguments: ** ** pSrb - pointer to the stream request block for properties ** ** Returns: ** ** Side Effects: none */
VOID VideoSetProperty( PHW_STREAM_REQUEST_BLOCK pSrb ) { PSTREAM_PROPERTY_DESCRIPTOR pSPD = pSrb->CommandData.PropertyInfo;
CDebugPrint(DebugLevelTrace,( CODECNAME ":--->VideoSetProperty(pSrb=%x)\n", pSrb));
if (IsEqualGUID (&KSPROPSETID_VBICodecFiltering, &pSPD->Property->Set)) { VideoStreamSetVBIFilteringProperty (pSrb); } else if( IsEqualGUID( &KSPROPSETID_Stream, &pSPD->Property->Set )) { pSrb->Status = STATUS_SUCCESS; } else { pSrb->Status = STATUS_NOT_IMPLEMENTED; }
CDebugPrint(DebugLevelTrace,( CODECNAME ":<---VideoSetProperty(pSrb=%x)\n", pSrb)); }
/*
** VideoSetState() ** ** Sets the current state of the requested stream ** ** Arguments: ** ** pSrb - pointer to the stream request block for properties ** ** Returns: ** ** Side Effects: none */
VOID VideoSetState( PHW_STREAM_REQUEST_BLOCK pSrb ) { PHW_DEVICE_EXTENSION pHwDevExt = ((PHW_DEVICE_EXTENSION)pSrb->HwDeviceExtension); PSTREAMEX pStrmEx = (PSTREAMEX)pSrb->StreamObject->HwStreamExtension; PHW_STREAM_REQUEST_BLOCK pCurrentSrb; int StreamNumber = (int)pSrb->StreamObject->StreamNumber;
CDebugPrint(DebugLevelTrace,( CODECNAME ":--->VideoSetState(pSrb=%x)\n", pSrb));
//
// For each stream, the following states are used:
//
// Stop: Absolute minimum resources are used. No outstanding IRPs.
// Acquire: KS only state that has no DirectShow correpondence
// Acquire needed resources.
// Pause: Getting ready to run. Allocate needed resources so that
// the eventual transition to Run is as fast as possible.
// Read SRBs will be queued at either the Stream class
// or in your driver (depending on when you send "ReadyForNext")
// Run: Streaming.
//
// Moving to Stop to Run always transitions through Pause.
//
// But since a client app could crash unexpectedly, drivers should handle
// the situation of having outstanding IRPs cancelled and open streams
// being closed WHILE THEY ARE STREAMING!
//
// Note that it is quite possible to transition repeatedly between states:
// Stop -> Pause -> Stop -> Pause -> Run -> Pause -> Run -> Pause -> Stop
//
switch (pSrb->CommandData.StreamState)
{ case KSSTATE_STOP:
//
// If transitioning to the stopped state, then complete any outstanding IRPs
//
while( QueueRemove( &pCurrentSrb, &pStrmEx->StreamDataSpinLock, &pStrmEx->StreamDataQueue )) { CDebugPrint(DebugLevelVerbose,( CODECNAME ": Cancelling %X\n", pCurrentSrb )); pCurrentSrb->Status = STATUS_CANCELLED; pCurrentSrb->CommandData.DataBufferArray->DataUsed = 0;
StreamClassStreamNotification( StreamRequestComplete, pCurrentSrb->StreamObject, pCurrentSrb ); } CDebugPrint( DebugLevelTrace, ( CODECNAME ": KSSTATE_STOP %u\n", StreamNumber )); break;
case KSSTATE_ACQUIRE:
//
// This is a KS only state, that has no correspondence in DirectShow
//
CDebugPrint( DebugLevelTrace, ( CODECNAME ": KSSTATE_ACQUIRE %u\n", StreamNumber )); break;
case KSSTATE_PAUSE:
//
// On a transition to pause from acquire, start our timer running.
//
if (pStrmEx->KSState == KSSTATE_ACQUIRE || pStrmEx->KSState == KSSTATE_STOP) {
// Remember the time at which the clock was started
pHwDevExt->QST_Start = VideoGetSystemTime();
// And initialize the last frame timestamp
pHwDevExt->QST_Now = pHwDevExt->QST_Start;
// Fireup the codec HERE in preparation for receiving data & requests.
// INSERT CODE HERE
} CDebugPrint( DebugLevelTrace, ( CODECNAME ": KSSTATE_PAUSE %u\n", StreamNumber )); break;
case KSSTATE_RUN:
//
// Begin Streaming.
//
// Remember the time at which the clock was started
pHwDevExt->QST_Start = VideoGetSystemTime();
// Zero the frame info, it should be reset when the first sample arrives.
RtlZeroMemory (&pStrmEx->FrameInfo, sizeof (pStrmEx->FrameInfo));
// Zero the last known picture numbers
pStrmEx->LastPictureNumber = 0; pHwDevExt->LastPictureNumber = 0;
// Reset the discontinuity flag
pStrmEx->fDiscontinuity = FALSE; CDebugPrint( DebugLevelTrace, ( CODECNAME ": KSSTATE_RUN %u\n", StreamNumber ));
break;
default: CDebugPrint( DebugLevelError, ( CODECNAME ": UNKNOWN STATE %u\n", StreamNumber )); CDEBUG_BREAK(); break;
} // end switch (pSrb->CommandData.StreamState)
//
// Remember the state of this stream
//
pStrmEx->KSState = pSrb->CommandData.StreamState;
CDebugPrint(DebugLevelTrace,( CODECNAME ":<---VideoSetState(pSrb=%x)\n", pSrb)); }
/*
** VideoGetState() ** ** Gets the current state of the requested stream ** ** Arguments: ** ** pSrb - pointer to the stream request block for properties ** ** Returns: ** ** Side Effects: none */
VOID VideoGetState( PHW_STREAM_REQUEST_BLOCK pSrb ) { PSTREAMEX pStrmEx = (PSTREAMEX)pSrb->StreamObject->HwStreamExtension;
CDebugPrint(DebugLevelTrace,(CODECNAME ":--->VideoGetState(pSrb=%x)\n", pSrb));
pSrb->CommandData.StreamState = pStrmEx->KSState; pSrb->ActualBytesTransferred = sizeof (KSSTATE);
// A very odd rule:
// When transitioning from stop to pause, DShow tries to preroll
// the graph. Capture sources can't preroll, and indicate this
// by returning VFW_S_CANT_CUE in user mode. To indicate this
// condition from drivers, they must return ERROR_NO_DATA_DETECTED
if (pStrmEx->KSState == KSSTATE_PAUSE) { pSrb->Status = STATUS_NO_DATA_DETECTED; }
CDebugPrint(DebugLevelTrace,(CODECNAME ":<---VideoGetState(pSrb=%x)=%d\n", pSrb, pStrmEx->KSState)); }
/*
** VideoStreamGetConnectionProperty() ** ** Gets the current state of the requested stream ** ** Arguments: ** ** pSrb - pointer to the stream request block for properties ** ** Returns: ** ** Side Effects: none */
VOID VideoStreamGetConnectionProperty( PHW_STREAM_REQUEST_BLOCK pSrb ) { PSTREAMEX pStrmEx = (PSTREAMEX)pSrb->StreamObject->HwStreamExtension; PHW_DEVICE_EXTENSION pHwDevExt = pStrmEx->pHwDevExt; PSTREAM_PROPERTY_DESCRIPTOR pSPD = pSrb->CommandData.PropertyInfo; ULONG Id = pSPD->Property->Id; // index of the property
int StreamNumber = ( int )pSrb->StreamObject->StreamNumber;
CDebugPrint(DebugLevelTrace, ( CODECNAME ":--->VideoStreamGetConnectionProperty(pSrb=%x)\n", pSrb));
pSrb->ActualBytesTransferred = 0;
switch (Id) { case KSPROPERTY_CONNECTION_ALLOCATORFRAMING: { PKSALLOCATOR_FRAMING Framing = (PKSALLOCATOR_FRAMING) pSPD->PropertyInfo; // PKS_DATARANGE_VIDEO_VBI pVBIFormat;
CDebugPrint(DebugLevelVerbose, ( CODECNAME ": VideoStreamGetConnectionProperty : KSPROPERTY_CONNECTION_ALLOCATORFRAMING %u\n", StreamNumber));
Framing->RequirementsFlags = KSALLOCATOR_REQUIREMENTF_PREFERENCES_ONLY | KSALLOCATOR_REQUIREMENTF_SYSTEM_MEMORY | KSALLOCATOR_REQUIREMENTF_INPLACE_MODIFIER; Framing->PoolType = NonPagedPool; Framing->FileAlignment = 0; // None OR FILE_QUAD_ALIGNMENT OR PAGE_SIZE - 1;
Framing->Reserved = 0;
switch( StreamNumber ) { case STREAM_VBI: Framing->Frames = 8; Framing->FrameSize = pStrmEx->OpenedFormat.SampleSize; break;
case STREAM_CC: if( CodecCompareGUIDsAndFormatSize( &pStrmEx->OpenedFormat, pHwDevExt->Streams[STREAM_CC].hwStreamInfo.StreamFormatsArray[0], FALSE )) { Framing->Frames = 60; Framing->FrameSize = CCSamples; } else if( CodecCompareGUIDsAndFormatSize( &pStrmEx->OpenedFormat, pHwDevExt->Streams[STREAM_CC].hwStreamInfo.StreamFormatsArray[1], FALSE )) { Framing->Frames = 8; Framing->FrameSize = pStrmEx->OpenedFormat.SampleSize; } else { CDebugPrint( DebugLevelError, ( CODECNAME ": VideoStreamGetConnectionProperty: Invalid Format\n" )); CDEBUG_BREAK(); } break; #ifdef CCINPUTPIN
case STREAM_CCINPUT: Framing->Frames = 60; Framing->FrameSize = CCSamples; break; #endif // CCINPUTPIN
default: CDebugPrint( DebugLevelError, ( CODECNAME ": VideoStreamGetConnectionProperty: Invalid Stream #\n" )); CDEBUG_BREAK();
} CDebugPrint( DebugLevelVerbose, ( CODECNAME ": Negotiated sample size is %d\n", Framing->FrameSize )); pSrb->ActualBytesTransferred = sizeof (KSALLOCATOR_FRAMING); break; }
default: pSrb->Status = STATUS_NOT_IMPLEMENTED; CDebugPrint(DebugLevelVerbose, ( CODECNAME ": VideoStreamGetConnectionProperty : Unknown Property Id=%d\n", Id)); CDEBUG_BREAK(); break; }
CDebugPrint(DebugLevelTrace, ( CODECNAME ":<---VideoStreamGetConnectionProperty(pSrb=%x)\n", pSrb)); }
/*
** VideoStreamGetVBIFilteringProperty() ** ** Gets the current state of the requested stream ** ** Arguments: ** ** pSrb - pointer to the stream request block for properties ** ** Returns: ** ** Side Effects: none */
VOID VideoStreamGetVBIFilteringProperty( PHW_STREAM_REQUEST_BLOCK pSrb ) { PSTREAMEX pStrmEx = (PSTREAMEX)pSrb->StreamObject->HwStreamExtension; PSTREAM_PROPERTY_DESCRIPTOR pSPD = pSrb->CommandData.PropertyInfo; ULONG Id = pSPD->Property->Id; // index of the property
LONG nBytes = pSPD->PropertyOutputSize - sizeof(KSPROPERTY); // size of data supplied
CDebugPrint(DebugLevelTrace, ( CODECNAME ":--->VideoStreamGetVBIFilteringProperty(pSrb=%x)\n", pSrb));
ASSERT (nBytes >= sizeof (LONG));
pSrb->ActualBytesTransferred = 0; switch (Id) { case KSPROPERTY_VBICODECFILTERING_SCANLINES_REQUESTED_BIT_ARRAY: { PKSPROPERTY_VBICODECFILTERING_SCANLINES_S Property = (PKSPROPERTY_VBICODECFILTERING_SCANLINES_S) pSPD->PropertyInfo;
nBytes = min( nBytes, sizeof( pStrmEx->ScanlinesRequested ) ); RtlCopyMemory( &Property->Scanlines, &pStrmEx->ScanlinesRequested, nBytes ); pSrb->ActualBytesTransferred = nBytes + sizeof(KSPROPERTY); CDebugPrint(DebugLevelVerbose, ( CODECNAME ": VideoStreamGetVBIFilteringProperty : KSPROPERTY_VBICODECFILTERING_SCANLINES_REQUESTED_BIT_ARRAY\n"));
break; }
case KSPROPERTY_VBICODECFILTERING_SCANLINES_DISCOVERED_BIT_ARRAY: { PKSPROPERTY_VBICODECFILTERING_SCANLINES_S Property = (PKSPROPERTY_VBICODECFILTERING_SCANLINES_S) pSPD->PropertyInfo;
CDebugPrint(DebugLevelVerbose, ( CODECNAME ": VideoStreamGetVBIFilteringProperty : KSPROPERTY_VBICODECFILTERING_SCANLINES_DISCOVERED_BIT_ARRAY\n")); nBytes = min( nBytes, sizeof( pStrmEx->ScanlinesDiscovered ) ); RtlCopyMemory( &Property->Scanlines, &pStrmEx->ScanlinesDiscovered, nBytes ); // Clear the data after the read so that it's always "fresh"
RtlZeroMemory( &pStrmEx->ScanlinesDiscovered, nBytes ); pSrb->ActualBytesTransferred = nBytes + sizeof(KSPROPERTY); break; }
case KSPROPERTY_VBICODECFILTERING_SUBSTREAMS_REQUESTED_BIT_ARRAY: { PKSPROPERTY_VBICODECFILTERING_CC_SUBSTREAMS_S Property = (PKSPROPERTY_VBICODECFILTERING_CC_SUBSTREAMS_S) pSPD->PropertyInfo;
nBytes = min( nBytes, sizeof( pStrmEx->SubstreamsRequested ) ); RtlCopyMemory( &Property->Substreams, &pStrmEx->SubstreamsRequested, nBytes ); pSrb->ActualBytesTransferred = nBytes + sizeof(KSPROPERTY); CDebugPrint(DebugLevelInfo, ( CODECNAME ": VideoStreamGetVBIFilteringProperty : KSPROPERTY_VBICODECFILTERING_SUBSTREAMS_REQUESTED_BIT_ARRAY %08x\n", Property->Substreams )); break; }
case KSPROPERTY_VBICODECFILTERING_SUBSTREAMS_DISCOVERED_BIT_ARRAY: { PKSPROPERTY_VBICODECFILTERING_CC_SUBSTREAMS_S Property = (PKSPROPERTY_VBICODECFILTERING_CC_SUBSTREAMS_S) pSPD->PropertyInfo;
CDebugPrint(DebugLevelVerbose, ( CODECNAME ": VideoStreamGetVBIFilteringProperty : KSPROPERTY_VBICODECFILTERING_SUBSTREAMS_DISCOVERED_BIT_ARRAY\n")); nBytes = min( nBytes, sizeof( pStrmEx->SubstreamsDiscovered ) ); RtlCopyMemory( &Property->Substreams, &pStrmEx->SubstreamsDiscovered, nBytes ); // Clear the data after the read so that it's always "fresh"
RtlZeroMemory( &pStrmEx->SubstreamsDiscovered, nBytes ); pSrb->ActualBytesTransferred = nBytes + sizeof(KSPROPERTY); break; }
case KSPROPERTY_VBICODECFILTERING_STATISTICS: { PKSPROPERTY_VBICODECFILTERING_STATISTICS_CC_PIN_S Property = (PKSPROPERTY_VBICODECFILTERING_STATISTICS_CC_PIN_S) pSPD->PropertyInfo;
CDebugPrint(DebugLevelVerbose, ( CODECNAME ": VideoStreamGetVBIFilteringProperty : KSPROPERTY_VBICODECFILTERING_STATISTICS_CC_PIN_S\n")); nBytes = min( nBytes, sizeof( pStrmEx->PinStats ) ); RtlCopyMemory( &Property->Statistics, &pStrmEx->PinStats, nBytes ); pSrb->ActualBytesTransferred = nBytes + sizeof(KSPROPERTY); break; }
default: pSrb->Status = STATUS_NOT_IMPLEMENTED; CDebugPrint(DebugLevelVerbose, ( CODECNAME ": VideoStreamGetVBIFilteringProperty : Unknown Property Id=%d\n", Id)); CDEBUG_BREAK(); break; } CDebugPrint(DebugLevelTrace, ( CODECNAME ":<---VideoStreamGetVBIFilteringProperty(pSrb=%x)\n", pSrb)); }
/*
** VideoStreamSetVBIFilteringProperty() ** ** Sets the current state of the requested stream ** ** Arguments: ** ** pSrb - pointer to the stream request block for properties ** ** Returns: ** ** Side Effects: none */
VOID VideoStreamSetVBIFilteringProperty( PHW_STREAM_REQUEST_BLOCK pSrb ) { PSTREAMEX pStrmEx = (PSTREAMEX)pSrb->StreamObject->HwStreamExtension; PSTREAM_PROPERTY_DESCRIPTOR pSPD = pSrb->CommandData.PropertyInfo; ULONG Id = pSPD->Property->Id; // index of the property
ULONG nBytes = pSPD->PropertyOutputSize - sizeof(KSPROPERTY); // size of data supplied
ASSERT (nBytes >= sizeof (LONG));
CDebugPrint(DebugLevelTrace, ( CODECNAME ":--->VideoStreamSetVBIFilteringProperty(pSrb=%x)\n", pSrb));
pSrb->ActualBytesTransferred = 0; switch (Id) { case KSPROPERTY_VBICODECFILTERING_SCANLINES_REQUESTED_BIT_ARRAY: { PKSPROPERTY_VBICODECFILTERING_SCANLINES_S Property = (PKSPROPERTY_VBICODECFILTERING_SCANLINES_S) pSPD->PropertyInfo;
CDebugPrint(DebugLevelVerbose, ( CODECNAME ": VideoStreamSetVBIFilteringProperty : KSPROPERTY_VBICODECFILTERING_SCANLINES_REQUESTED_BIT_ARRAY\n")); nBytes = min( nBytes, sizeof( pStrmEx->ScanlinesRequested ) ); RtlCopyMemory( &pStrmEx->ScanlinesRequested, &Property->Scanlines, nBytes ); pSrb->ActualBytesTransferred = nBytes + sizeof(KSPROPERTY); break; } #ifdef SETDISCOVERED
case KSPROPERTY_VBICODECFILTERING_SCANLINES_DISCOVERED_BIT_ARRAY: { PKSPROPERTY_VBICODECFILTERING_SCANLINES_S Property = (PKSPROPERTY_VBICODECFILTERING_SCANLINES_S) pSPD->PropertyInfo;
CDebugPrint(DebugLevelVerbose, ( CODECNAME ": VideoStreamSetVBIFilteringProperty : KSPROPERTY_VBICODECFILTERING_SCANLINES_DISCOVERED_BIT_ARRAY\n")); nBytes = min( nBytes, sizeof(pStrmEx->ScanlinesDiscovered ) ); RtlCopyMemory( &pStrmEx->ScanlinesDiscovered, &Property->Scanlines, nBytes ); pSrb->ActualBytesTransferred = nBytes + sizeof(KSPROPERTY); break; } #endif // SETDISCOVERED
case KSPROPERTY_VBICODECFILTERING_SUBSTREAMS_REQUESTED_BIT_ARRAY: { PKSPROPERTY_VBICODECFILTERING_CC_SUBSTREAMS_S Property = (PKSPROPERTY_VBICODECFILTERING_CC_SUBSTREAMS_S) pSPD->PropertyInfo;
nBytes = min( nBytes, sizeof(pStrmEx->SubstreamsRequested ) ); RtlCopyMemory( &pStrmEx->SubstreamsRequested, &Property->Substreams, nBytes ); pSrb->ActualBytesTransferred = nBytes + sizeof(KSPROPERTY); CDebugPrint(DebugLevelInfo, ( CODECNAME ": VideoStreamSetVBIFilteringProperty : KSPROPERTY_VBICODECFILTERING_SUBSTREAMS_REQUESTED_BIT_ARRAY %08x\n", pStrmEx->SubstreamsRequested.SubstreamMask));
break; } #ifdef SETDISCOVERED
case KSPROPERTY_VBICODECFILTERING_SUBSTREAMS_DISCOVERED_BIT_ARRAY: { PKSPROPERTY_VBICODECFILTERING_CC_SUBSTREAMS_S Property = (PKSPROPERTY_VBICODECFILTERING_CC_SUBSTREAMS_S) pSPD->PropertyInfo;
CDebugPrint(DebugLevelVerbose, ( CODECNAME ": VideoStreamSetVBIFilteringProperty : KSPROPERTY_VBICODECFILTERING_SUBSTREAMS_DISCOVERED_BIT_ARRAY\n")); nBytes = min( nBytes, sizeof(pStrmEx->SubstreamsDiscovered ) ); RtlCopyMemory( &pStrmEx->SubstreamsDiscovered, &Property->Substreams, nBytes ); pSrb->ActualBytesTransferred = nBytes + sizeof(KSPROPERTY); break; } #endif // SETDISCOVERED
case KSPROPERTY_VBICODECFILTERING_STATISTICS: { PKSPROPERTY_VBICODECFILTERING_STATISTICS_CC_PIN_S Property = (PKSPROPERTY_VBICODECFILTERING_STATISTICS_CC_PIN_S) pSPD->PropertyInfo;
CDebugPrint(DebugLevelVerbose, ( CODECNAME ": VideoStreamSetVBIFilteringProperty : KSPROPERTY_VBICODECFILTERING_STATISTICS\n")); nBytes = min( nBytes, sizeof( pStrmEx->PinStats ) ); RtlCopyMemory( &pStrmEx->PinStats, &Property->Statistics, nBytes ); pSrb->ActualBytesTransferred = nBytes + sizeof(KSPROPERTY); break; }
default: pSrb->Status = STATUS_NOT_IMPLEMENTED; CDebugPrint(DebugLevelVerbose, ( CODECNAME ": VideoStreamSetVBIFilteringProperty : Unknown Property Id=%d\n", Id)); CDEBUG_BREAK(); break; } CDebugPrint(DebugLevelTrace, ( CODECNAME ":<---VideoStreamSetVBIFilteringProperty(pSrb=%x)\n", pSrb)); }
/*
** GetSystemTime () ** ** Returns the system time in 100 nS units ** ** Arguments: ** ** Returns: ** ** Side Effects: none */
ULONGLONG VideoGetSystemTime() { ULONGLONG ticks; ULONGLONG rate;
CDebugPrint(DebugLevelTrace,( CODECNAME ":--->VideoGetSystemTime()\n"));
ticks = (ULONGLONG)KeQueryPerformanceCounter((PLARGE_INTEGER)&rate).QuadPart;
//
// convert from ticks to 100ns clock
//
ticks = (ticks & 0xFFFFFFFF00000000) / rate * 10000000 + (ticks & 0x00000000FFFFFFFF) * 10000000 / rate;
CDebugPrint(DebugLevelTrace,( CODECNAME ":<---VideoGetSystemTime()\n"));
return(ticks); }
//==========================================================================;
// Clock Handling Routines
//==========================================================================;
/*
** VideoIndicateMasterClock () ** ** This function is used to provide us with a handle to the clock to use. ** ** Arguments: ** ** pSrb - pointer to the stream request block for properties ** ** Returns: ** ** Side Effects: none */
VOID VideoIndicateMasterClock( PHW_STREAM_REQUEST_BLOCK pSrb ) { PSTREAMEX pStrmEx = (PSTREAMEX)pSrb->StreamObject->HwStreamExtension;
CDebugPrint(DebugLevelTrace,( CODECNAME ":--->VideoIndicateMasterClock(pSrb=%x)\n", pSrb));
pStrmEx->hClock = pSrb->CommandData.MasterClockHandle;
CDebugPrint(DebugLevelTrace,( CODECNAME ":<---VideoIndicateMasterClock(pSrb=%x)\n", pSrb)); }
|