431 lines
8.9 KiB
C
431 lines
8.9 KiB
C
/*
|
|
* Implementation of IMediaFilter for FilterGraph.
|
|
*
|
|
* FIXME - stub.
|
|
*
|
|
* hidenori@a2.ctktv.ne.jp
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "windef.h"
|
|
#include "winbase.h"
|
|
#include "wingdi.h"
|
|
#include "winuser.h"
|
|
#include "winerror.h"
|
|
#include "strmif.h"
|
|
#include "control.h"
|
|
#include "uuids.h"
|
|
#include "vfwmsgs.h"
|
|
#include "evcode.h"
|
|
|
|
#include "debugtools.h"
|
|
DEFAULT_DEBUG_CHANNEL(quartz);
|
|
|
|
#include "quartz_private.h"
|
|
#include "fgraph.h"
|
|
|
|
|
|
#define WINE_QUARTZ_POLL_INTERVAL 10
|
|
|
|
/*****************************************************************************/
|
|
|
|
static
|
|
HRESULT CFilterGraph_PollGraphState(
|
|
CFilterGraph* This,
|
|
FILTER_STATE* pState)
|
|
{
|
|
HRESULT hr;
|
|
QUARTZ_CompListItem* pItem;
|
|
IBaseFilter* pFilter;
|
|
|
|
hr = S_OK;
|
|
|
|
EnterCriticalSection( &This->m_csGraphState );
|
|
QUARTZ_CompList_Lock( This->m_pFilterList );
|
|
|
|
pItem = QUARTZ_CompList_GetFirst( This->m_pFilterList );
|
|
|
|
while ( pItem != NULL )
|
|
{
|
|
pFilter = (IBaseFilter*)QUARTZ_CompList_GetItemPtr( pItem );
|
|
hr = IBaseFilter_GetState( pFilter, (DWORD)0, pState );
|
|
if ( hr != S_OK )
|
|
break;
|
|
|
|
pItem = QUARTZ_CompList_GetNext( This->m_pFilterList, pItem );
|
|
}
|
|
|
|
QUARTZ_CompList_Unlock( This->m_pFilterList );
|
|
LeaveCriticalSection( &This->m_csGraphState );
|
|
|
|
TRACE( "returns %08lx, state %d\n",
|
|
hr, *pState );
|
|
|
|
return hr;
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
|
|
static HRESULT WINAPI
|
|
IMediaFilter_fnQueryInterface(IMediaFilter* iface,REFIID riid,void** ppobj)
|
|
{
|
|
CFilterGraph_THIS(iface,mediafilter);
|
|
|
|
TRACE("(%p)->()\n",This);
|
|
|
|
return IUnknown_QueryInterface(This->unk.punkControl,riid,ppobj);
|
|
}
|
|
|
|
static ULONG WINAPI
|
|
IMediaFilter_fnAddRef(IMediaFilter* iface)
|
|
{
|
|
CFilterGraph_THIS(iface,mediafilter);
|
|
|
|
TRACE("(%p)->()\n",This);
|
|
|
|
return IUnknown_AddRef(This->unk.punkControl);
|
|
}
|
|
|
|
static ULONG WINAPI
|
|
IMediaFilter_fnRelease(IMediaFilter* iface)
|
|
{
|
|
CFilterGraph_THIS(iface,mediafilter);
|
|
|
|
TRACE("(%p)->()\n",This);
|
|
|
|
return IUnknown_Release(This->unk.punkControl);
|
|
}
|
|
|
|
|
|
static HRESULT WINAPI
|
|
IMediaFilter_fnGetClassID(IMediaFilter* iface,CLSID* pclsid)
|
|
{
|
|
CFilterGraph_THIS(iface,mediafilter);
|
|
|
|
TRACE("(%p)->()\n",This);
|
|
|
|
return IPersist_GetClassID(
|
|
CFilterGraph_IPersist(This),pclsid);
|
|
}
|
|
|
|
static HRESULT WINAPI
|
|
IMediaFilter_fnStop(IMediaFilter* iface)
|
|
{
|
|
CFilterGraph_THIS(iface,mediafilter);
|
|
HRESULT hr;
|
|
HRESULT hrFilter;
|
|
QUARTZ_CompListItem* pItem;
|
|
IBaseFilter* pFilter;
|
|
|
|
TRACE("(%p)->()\n",This);
|
|
|
|
hr = S_OK;
|
|
|
|
EnterCriticalSection( &This->m_csGraphState );
|
|
|
|
if ( This->m_stateGraph != State_Stopped )
|
|
{
|
|
/* IDistributorNotify_Stop() */
|
|
|
|
QUARTZ_CompList_Lock( This->m_pFilterList );
|
|
|
|
pItem = QUARTZ_CompList_GetFirst( This->m_pFilterList );
|
|
|
|
while ( pItem != NULL )
|
|
{
|
|
pFilter = (IBaseFilter*)QUARTZ_CompList_GetItemPtr( pItem );
|
|
hrFilter = IBaseFilter_Stop( pFilter );
|
|
if ( hrFilter != S_OK )
|
|
{
|
|
if ( SUCCEEDED(hr) )
|
|
hr = hrFilter;
|
|
}
|
|
|
|
pItem = QUARTZ_CompList_GetNext( This->m_pFilterList, pItem );
|
|
}
|
|
|
|
QUARTZ_CompList_Unlock( This->m_pFilterList );
|
|
|
|
This->m_stateGraph = State_Stopped;
|
|
}
|
|
|
|
LeaveCriticalSection( &This->m_csGraphState );
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI
|
|
IMediaFilter_fnPause(IMediaFilter* iface)
|
|
{
|
|
CFilterGraph_THIS(iface,mediafilter);
|
|
HRESULT hr;
|
|
HRESULT hrFilter;
|
|
QUARTZ_CompListItem* pItem;
|
|
IBaseFilter* pFilter;
|
|
|
|
TRACE("(%p)->()\n",This);
|
|
|
|
hr = S_OK;
|
|
|
|
EnterCriticalSection( &This->m_csGraphState );
|
|
|
|
if ( This->m_stateGraph != State_Paused )
|
|
{
|
|
/* IDistributorNotify_Pause() */
|
|
|
|
QUARTZ_CompList_Lock( This->m_pFilterList );
|
|
|
|
pItem = QUARTZ_CompList_GetFirst( This->m_pFilterList );
|
|
|
|
while ( pItem != NULL )
|
|
{
|
|
pFilter = (IBaseFilter*)QUARTZ_CompList_GetItemPtr( pItem );
|
|
hrFilter = IBaseFilter_Pause( pFilter );
|
|
if ( hrFilter != S_OK )
|
|
{
|
|
if ( SUCCEEDED(hr) )
|
|
hr = hrFilter;
|
|
}
|
|
|
|
pItem = QUARTZ_CompList_GetNext( This->m_pFilterList, pItem );
|
|
}
|
|
|
|
QUARTZ_CompList_Unlock( This->m_pFilterList );
|
|
|
|
This->m_stateGraph = State_Paused;
|
|
}
|
|
|
|
LeaveCriticalSection( &This->m_csGraphState );
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI
|
|
IMediaFilter_fnRun(IMediaFilter* iface,REFERENCE_TIME rtStart)
|
|
{
|
|
CFilterGraph_THIS(iface,mediafilter);
|
|
HRESULT hr;
|
|
HRESULT hrFilter;
|
|
QUARTZ_CompListItem* pItem;
|
|
IBaseFilter* pFilter;
|
|
IReferenceClock* pClock;
|
|
|
|
TRACE("(%p)->()\n",This);
|
|
|
|
EnterCriticalSection( &This->m_csGraphState );
|
|
|
|
if ( This->m_stateGraph == State_Stopped )
|
|
{
|
|
hr = IMediaFilter_Pause(iface);
|
|
if ( FAILED(hr) )
|
|
goto end;
|
|
}
|
|
|
|
/* handle the special time. */
|
|
if ( rtStart == (REFERENCE_TIME)0 )
|
|
{
|
|
hr = IMediaFilter_GetSyncSource(iface,&pClock);
|
|
if ( hr == S_OK && pClock != NULL )
|
|
{
|
|
IReferenceClock_GetTime(pClock,&rtStart);
|
|
IReferenceClock_Release(pClock);
|
|
}
|
|
}
|
|
|
|
hr = NOERROR;
|
|
|
|
if ( This->m_stateGraph != State_Running )
|
|
{
|
|
/* IDistributorNotify_Run() */
|
|
|
|
QUARTZ_CompList_Lock( This->m_pFilterList );
|
|
|
|
pItem = QUARTZ_CompList_GetFirst( This->m_pFilterList );
|
|
|
|
while ( pItem != NULL )
|
|
{
|
|
pFilter = (IBaseFilter*)QUARTZ_CompList_GetItemPtr( pItem );
|
|
hrFilter = IBaseFilter_Run( pFilter, rtStart );
|
|
if ( hrFilter != S_OK )
|
|
{
|
|
if ( SUCCEEDED(hr) )
|
|
hr = hrFilter;
|
|
}
|
|
|
|
pItem = QUARTZ_CompList_GetNext( This->m_pFilterList, pItem );
|
|
}
|
|
|
|
QUARTZ_CompList_Unlock( This->m_pFilterList );
|
|
|
|
This->m_stateGraph = State_Running;
|
|
}
|
|
|
|
end:
|
|
LeaveCriticalSection( &This->m_csGraphState );
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI
|
|
IMediaFilter_fnGetState(IMediaFilter* iface,DWORD dwTimeOut,FILTER_STATE* pState)
|
|
{
|
|
CFilterGraph_THIS(iface,mediafilter);
|
|
HRESULT hr;
|
|
DWORD dwTickStart;
|
|
DWORD dwTickUsed;
|
|
|
|
TRACE("(%p)->(%p)\n",This,pState);
|
|
if ( pState == NULL )
|
|
return E_POINTER;
|
|
|
|
dwTickStart = GetTickCount();
|
|
|
|
while ( 1 )
|
|
{
|
|
hr = CFilterGraph_PollGraphState( This, pState );
|
|
if ( hr != VFW_S_STATE_INTERMEDIATE )
|
|
break;
|
|
if ( dwTimeOut == 0 )
|
|
break;
|
|
|
|
Sleep( (dwTimeOut >= WINE_QUARTZ_POLL_INTERVAL) ?
|
|
WINE_QUARTZ_POLL_INTERVAL : dwTimeOut );
|
|
|
|
dwTickUsed = GetTickCount() - dwTickStart;
|
|
|
|
dwTickStart += dwTickUsed;
|
|
if ( dwTimeOut <= dwTickUsed )
|
|
dwTimeOut = 0;
|
|
else
|
|
dwTimeOut -= dwTickUsed;
|
|
}
|
|
|
|
EnterCriticalSection( &This->m_csGraphState );
|
|
*pState = This->m_stateGraph;
|
|
LeaveCriticalSection( &This->m_csGraphState );
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI
|
|
IMediaFilter_fnSetSyncSource(IMediaFilter* iface,IReferenceClock* pobjClock)
|
|
{
|
|
CFilterGraph_THIS(iface,mediafilter);
|
|
QUARTZ_CompListItem* pItem;
|
|
IBaseFilter* pFilter;
|
|
HRESULT hr = NOERROR;
|
|
HRESULT hrCur;
|
|
|
|
TRACE("(%p)->(%p)\n",This,pobjClock);
|
|
|
|
/* IDistributorNotify_SetSyncSource() */
|
|
|
|
EnterCriticalSection( &This->m_csClock );
|
|
QUARTZ_CompList_Lock( This->m_pFilterList );
|
|
|
|
if ( This->m_pClock != NULL )
|
|
{
|
|
IReferenceClock_Release(This->m_pClock);
|
|
This->m_pClock = NULL;
|
|
}
|
|
|
|
This->m_pClock = pobjClock;
|
|
if ( pobjClock != NULL )
|
|
IReferenceClock_AddRef( pobjClock );
|
|
|
|
pItem = QUARTZ_CompList_GetFirst( This->m_pFilterList );
|
|
while ( pItem != NULL )
|
|
{
|
|
pFilter = (IBaseFilter*)QUARTZ_CompList_GetItemPtr( pItem );
|
|
hrCur = IBaseFilter_SetSyncSource(pFilter,pobjClock);
|
|
if ( FAILED(hrCur) )
|
|
hr = hrCur;
|
|
pItem = QUARTZ_CompList_GetNext( This->m_pFilterList, pItem );
|
|
}
|
|
|
|
QUARTZ_CompList_Unlock( This->m_pFilterList );
|
|
|
|
IMediaEventSink_Notify(CFilterGraph_IMediaEventSink(This),
|
|
EC_CLOCK_CHANGED, 0, 0);
|
|
|
|
LeaveCriticalSection( &This->m_csClock );
|
|
|
|
TRACE( "hr = %08lx\n", hr );
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI
|
|
IMediaFilter_fnGetSyncSource(IMediaFilter* iface,IReferenceClock** ppobjClock)
|
|
{
|
|
CFilterGraph_THIS(iface,mediafilter);
|
|
HRESULT hr = VFW_E_NO_CLOCK;
|
|
|
|
TRACE("(%p)->(%p)\n",This,ppobjClock);
|
|
|
|
if ( ppobjClock == NULL )
|
|
return E_POINTER;
|
|
|
|
EnterCriticalSection( &This->m_csClock );
|
|
*ppobjClock = This->m_pClock;
|
|
if ( This->m_pClock != NULL )
|
|
{
|
|
hr = NOERROR;
|
|
IReferenceClock_AddRef( This->m_pClock );
|
|
}
|
|
LeaveCriticalSection( &This->m_csClock );
|
|
|
|
TRACE( "hr = %08lx\n", hr );
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
static ICOM_VTABLE(IMediaFilter) imediafilter =
|
|
{
|
|
ICOM_MSVTABLE_COMPAT_DummyRTTIVALUE
|
|
/* IUnknown fields */
|
|
IMediaFilter_fnQueryInterface,
|
|
IMediaFilter_fnAddRef,
|
|
IMediaFilter_fnRelease,
|
|
/* IPersist fields */
|
|
IMediaFilter_fnGetClassID,
|
|
/* IMediaFilter fields */
|
|
IMediaFilter_fnStop,
|
|
IMediaFilter_fnPause,
|
|
IMediaFilter_fnRun,
|
|
IMediaFilter_fnGetState,
|
|
IMediaFilter_fnSetSyncSource,
|
|
IMediaFilter_fnGetSyncSource,
|
|
};
|
|
|
|
HRESULT CFilterGraph_InitIMediaFilter( CFilterGraph* pfg )
|
|
{
|
|
TRACE("(%p)\n",pfg);
|
|
|
|
ICOM_VTBL(&pfg->mediafilter) = &imediafilter;
|
|
|
|
InitializeCriticalSection( &pfg->m_csGraphState );
|
|
InitializeCriticalSection( &pfg->m_csClock );
|
|
pfg->m_stateGraph = State_Stopped;
|
|
pfg->m_pClock = NULL;
|
|
|
|
return NOERROR;
|
|
}
|
|
|
|
void CFilterGraph_UninitIMediaFilter( CFilterGraph* pfg )
|
|
{
|
|
TRACE("(%p)\n",pfg);
|
|
|
|
if ( pfg->m_pClock != NULL )
|
|
{
|
|
IReferenceClock_Release( pfg->m_pClock );
|
|
pfg->m_pClock = NULL;
|
|
}
|
|
|
|
DeleteCriticalSection( &pfg->m_csGraphState );
|
|
DeleteCriticalSection( &pfg->m_csClock );
|
|
}
|