251 lines
6.8 KiB
C
251 lines
6.8 KiB
C
/*
|
|
* Generic Implementation of IBaseFilter Interface
|
|
*
|
|
* Copyright 2010 Aric Stewart, CodeWeavers
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
*/
|
|
|
|
#include "strmbase_private.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(strmbase);
|
|
|
|
static inline BaseFilter *impl_from_IBaseFilter(IBaseFilter *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, BaseFilter, IBaseFilter_iface);
|
|
}
|
|
|
|
HRESULT WINAPI BaseFilterImpl_QueryInterface(IBaseFilter *iface, REFIID iid, void **out)
|
|
{
|
|
BaseFilter *filter = impl_from_IBaseFilter(iface);
|
|
HRESULT hr;
|
|
|
|
TRACE("iface %p, iid %s, out %p.\n", iface, debugstr_guid(iid), out);
|
|
|
|
*out = NULL;
|
|
|
|
if (filter->pFuncsTable->filter_query_interface
|
|
&& SUCCEEDED(hr = filter->pFuncsTable->filter_query_interface(filter, iid, out)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
if (IsEqualIID(iid, &IID_IUnknown)
|
|
|| IsEqualIID(iid, &IID_IPersist)
|
|
|| IsEqualIID(iid, &IID_IMediaFilter)
|
|
|| IsEqualIID(iid, &IID_IBaseFilter))
|
|
{
|
|
*out = iface;
|
|
IBaseFilter_AddRef(iface);
|
|
return S_OK;
|
|
}
|
|
|
|
WARN("%s not implemented, returning E_NOINTERFACE.\n", debugstr_guid(iid));
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
ULONG WINAPI BaseFilterImpl_AddRef(IBaseFilter * iface)
|
|
{
|
|
BaseFilter *This = impl_from_IBaseFilter(iface);
|
|
ULONG refCount = InterlockedIncrement(&This->refCount);
|
|
|
|
TRACE("(%p)->() AddRef from %d\n", This, refCount - 1);
|
|
|
|
return refCount;
|
|
}
|
|
|
|
ULONG WINAPI BaseFilterImpl_Release(IBaseFilter *iface)
|
|
{
|
|
BaseFilter *This = impl_from_IBaseFilter(iface);
|
|
ULONG refCount = InterlockedDecrement(&This->refCount);
|
|
|
|
TRACE("(%p)->() Release from %d\n", This, refCount + 1);
|
|
|
|
if (!refCount)
|
|
This->pFuncsTable->filter_destroy(This);
|
|
|
|
return refCount;
|
|
}
|
|
|
|
HRESULT WINAPI BaseFilterImpl_GetClassID(IBaseFilter * iface, CLSID * pClsid)
|
|
{
|
|
BaseFilter *This = impl_from_IBaseFilter(iface);
|
|
TRACE("(%p)->(%p)\n", This, pClsid);
|
|
|
|
*pClsid = This->clsid;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT WINAPI BaseFilterImpl_GetState(IBaseFilter * iface, DWORD dwMilliSecsTimeout, FILTER_STATE *pState )
|
|
{
|
|
BaseFilter *This = impl_from_IBaseFilter(iface);
|
|
TRACE("(%p)->(%d, %p)\n", This, dwMilliSecsTimeout, pState);
|
|
|
|
EnterCriticalSection(&This->csFilter);
|
|
{
|
|
*pState = This->state;
|
|
}
|
|
LeaveCriticalSection(&This->csFilter);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT WINAPI BaseFilterImpl_SetSyncSource(IBaseFilter * iface, IReferenceClock *pClock)
|
|
{
|
|
BaseFilter *This = impl_from_IBaseFilter(iface);
|
|
TRACE("(%p)->(%p)\n", This, pClock);
|
|
|
|
EnterCriticalSection(&This->csFilter);
|
|
{
|
|
if (This->pClock)
|
|
IReferenceClock_Release(This->pClock);
|
|
This->pClock = pClock;
|
|
if (This->pClock)
|
|
IReferenceClock_AddRef(This->pClock);
|
|
}
|
|
LeaveCriticalSection(&This->csFilter);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT WINAPI BaseFilterImpl_GetSyncSource(IBaseFilter * iface, IReferenceClock **ppClock)
|
|
{
|
|
BaseFilter *This = impl_from_IBaseFilter(iface);
|
|
TRACE("(%p)->(%p)\n", This, ppClock);
|
|
|
|
EnterCriticalSection(&This->csFilter);
|
|
{
|
|
*ppClock = This->pClock;
|
|
if (This->pClock)
|
|
IReferenceClock_AddRef(This->pClock);
|
|
}
|
|
LeaveCriticalSection(&This->csFilter);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT WINAPI BaseFilterImpl_EnumPins(IBaseFilter *iface, IEnumPins **enum_pins)
|
|
{
|
|
BaseFilter *filter = impl_from_IBaseFilter(iface);
|
|
|
|
TRACE("iface %p, enum_pins %p.\n", iface, enum_pins);
|
|
|
|
return enum_pins_create(filter, enum_pins);
|
|
}
|
|
|
|
HRESULT WINAPI BaseFilterImpl_FindPin(IBaseFilter *iface, const WCHAR *id, IPin **ret)
|
|
{
|
|
BaseFilter *This = impl_from_IBaseFilter(iface);
|
|
unsigned int i;
|
|
PIN_INFO info;
|
|
HRESULT hr;
|
|
IPin *pin;
|
|
|
|
TRACE("(%p)->(%s, %p)\n", This, debugstr_w(id), ret);
|
|
|
|
for (i = 0; (pin = This->pFuncsTable->filter_get_pin(This, i)); ++i)
|
|
{
|
|
hr = IPin_QueryPinInfo(pin, &info);
|
|
if (FAILED(hr))
|
|
{
|
|
IPin_Release(pin);
|
|
return hr;
|
|
}
|
|
if (info.pFilter) IBaseFilter_Release(info.pFilter);
|
|
|
|
if (!lstrcmpW(id, info.achName))
|
|
{
|
|
*ret = pin;
|
|
return S_OK;
|
|
}
|
|
|
|
IPin_Release(pin);
|
|
}
|
|
|
|
return VFW_E_NOT_FOUND;
|
|
}
|
|
|
|
HRESULT WINAPI BaseFilterImpl_QueryFilterInfo(IBaseFilter * iface, FILTER_INFO *pInfo)
|
|
{
|
|
BaseFilter *This = impl_from_IBaseFilter(iface);
|
|
TRACE("(%p)->(%p)\n", This, pInfo);
|
|
|
|
lstrcpyW(pInfo->achName, This->filterInfo.achName);
|
|
pInfo->pGraph = This->filterInfo.pGraph;
|
|
|
|
if (pInfo->pGraph)
|
|
IFilterGraph_AddRef(pInfo->pGraph);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT WINAPI BaseFilterImpl_JoinFilterGraph(IBaseFilter * iface, IFilterGraph *pGraph, LPCWSTR pName )
|
|
{
|
|
BaseFilter *This = impl_from_IBaseFilter(iface);
|
|
|
|
TRACE("(%p)->(%p, %s)\n", This, pGraph, debugstr_w(pName));
|
|
|
|
EnterCriticalSection(&This->csFilter);
|
|
{
|
|
if (pName)
|
|
lstrcpyW(This->filterInfo.achName, pName);
|
|
else
|
|
*This->filterInfo.achName = '\0';
|
|
This->filterInfo.pGraph = pGraph; /* NOTE: do NOT increase ref. count */
|
|
}
|
|
LeaveCriticalSection(&This->csFilter);
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT WINAPI BaseFilterImpl_QueryVendorInfo(IBaseFilter * iface, LPWSTR *pVendorInfo)
|
|
{
|
|
TRACE("(%p)->(%p)\n", iface, pVendorInfo);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
VOID WINAPI BaseFilterImpl_IncrementPinVersion(BaseFilter *filter)
|
|
{
|
|
InterlockedIncrement(&filter->pin_version);
|
|
}
|
|
|
|
void BaseFilter_Init(BaseFilter *This, const IBaseFilterVtbl *Vtbl, const CLSID *pClsid,
|
|
DWORD_PTR DebugInfo, const BaseFilterFuncTable *pBaseFuncsTable)
|
|
{
|
|
This->IBaseFilter_iface.lpVtbl = Vtbl;
|
|
This->refCount = 1;
|
|
InitializeCriticalSection(&This->csFilter);
|
|
This->state = State_Stopped;
|
|
This->rtStreamStart = 0;
|
|
This->pClock = NULL;
|
|
ZeroMemory(&This->filterInfo, sizeof(FILTER_INFO));
|
|
This->clsid = *pClsid;
|
|
This->csFilter.DebugInfo->Spare[0] = DebugInfo;
|
|
This->pin_version = 1;
|
|
|
|
This->pFuncsTable = pBaseFuncsTable;
|
|
}
|
|
|
|
void strmbase_filter_cleanup(BaseFilter *This)
|
|
{
|
|
if (This->pClock)
|
|
IReferenceClock_Release(This->pClock);
|
|
|
|
This->IBaseFilter_iface.lpVtbl = NULL;
|
|
This->csFilter.DebugInfo->Spare[0] = 0;
|
|
DeleteCriticalSection(&This->csFilter);
|
|
}
|