| /* |
| * Implementation of IFilterGraph and related interfaces |
| * + IGraphVersion |
| * |
| * FIXME - create a thread to process some methods correctly. |
| * |
| * FIXME - ReconnectEx |
| * FIXME - process Pause/Stop asynchronously and notify when completed. |
| * |
| * |
| * Copyright (C) Hidenori TAKESHIMA <hidenori@a2.ctktv.ne.jp> |
| * |
| * 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
| */ |
| |
| #include "config.h" |
| |
| #include "windef.h" |
| #include "winbase.h" |
| #include "wingdi.h" |
| #include "winuser.h" |
| #include "winerror.h" |
| #include "winreg.h" |
| #include "strmif.h" |
| #include "control.h" |
| #include "uuids.h" |
| #include "vfwmsgs.h" |
| #include "evcode.h" |
| |
| #include "wine/debug.h" |
| WINE_DEFAULT_DEBUG_CHANNEL(quartz); |
| |
| #include "quartz_private.h" |
| #include "fgraph.h" |
| #include "enumunk.h" |
| #include "sysclock.h" |
| #include "regsvr.h" |
| |
| |
| #ifndef NUMELEMS |
| #define NUMELEMS(elem) (sizeof(elem)/sizeof(elem[0])) |
| #endif /* NUMELEMS */ |
| |
| static HRESULT CFilterGraph_DisconnectAllPins( IBaseFilter* pFilter ) |
| { |
| IEnumPins* pEnum = NULL; |
| IPin* pPin; |
| IPin* pConnTo; |
| ULONG cFetched; |
| HRESULT hr; |
| |
| hr = IBaseFilter_EnumPins( pFilter, &pEnum ); |
| if ( FAILED(hr) ) |
| return hr; |
| if ( pEnum == NULL ) |
| return E_FAIL; |
| |
| while ( 1 ) |
| { |
| pPin = NULL; |
| cFetched = 0; |
| hr = IEnumPins_Next( pEnum, 1, &pPin, &cFetched ); |
| if ( FAILED(hr) ) |
| break; |
| if ( hr != NOERROR || pPin == NULL || cFetched != 1 ) |
| { |
| hr = NOERROR; |
| break; |
| } |
| |
| pConnTo = NULL; |
| hr = IPin_ConnectedTo(pPin,&pConnTo); |
| if ( hr == NOERROR && pConnTo != NULL ) |
| { |
| IPin_Disconnect(pPin); |
| IPin_Disconnect(pConnTo); |
| IPin_Release(pConnTo); |
| } |
| |
| IPin_Release( pPin ); |
| } |
| |
| IEnumPins_Release( pEnum ); |
| |
| return hr; |
| } |
| |
| |
| static HRESULT CFilterGraph_GraphChanged( CFilterGraph* This ) |
| { |
| /* IDistributorNotify_NotifyGraphChange() */ |
| |
| This->m_lGraphVersion ++; |
| |
| return NOERROR; |
| } |
| |
| /*************************************************************************** |
| * |
| * CFilterGraph internal methods for IFilterGraph2::AddSourceFilter(). |
| * |
| */ |
| |
| static HRESULT QUARTZ_PeekFile( |
| const WCHAR* pwszFileName, |
| BYTE* pData, DWORD cbData, DWORD* pcbRead ) |
| { |
| HANDLE hFile; |
| HRESULT hr = E_FAIL; |
| |
| *pcbRead = 0; |
| hFile = CreateFileW( pwszFileName, |
| GENERIC_READ, FILE_SHARE_READ, |
| NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, (HANDLE)NULL ); |
| if ( hFile == INVALID_HANDLE_VALUE ) |
| return E_FAIL; |
| if ( ReadFile( hFile, pData, cbData, pcbRead, NULL ) ) |
| hr = NOERROR; |
| CloseHandle( hFile ); |
| |
| return hr; |
| } |
| |
| |
| static const WCHAR* skip_space(const WCHAR* pwsz) |
| { |
| if ( pwsz == NULL ) return NULL; |
| while ( *pwsz == (WCHAR)' ' ) pwsz++; |
| return pwsz; |
| } |
| |
| static const WCHAR* get_dword(const WCHAR* pwsz,DWORD* pdw) |
| { |
| DWORD dw = 0; |
| |
| *pdw = 0; |
| if ( pwsz == NULL ) return NULL; |
| while ( *pwsz >= (WCHAR)'0' && *pwsz <= (WCHAR)'9' ) |
| { |
| dw = dw * 10 + (DWORD)(*pwsz-(WCHAR)'0'); |
| pwsz ++; |
| } |
| *pdw = dw; |
| return pwsz; |
| } |
| |
| static int wchar_to_hex(WCHAR wch) |
| { |
| if ( wch >= (WCHAR)'0' && wch <= (WCHAR)'9' ) |
| return (int)(wch - (WCHAR)'0'); |
| if ( wch >= (WCHAR)'A' && wch <= (WCHAR)'F' ) |
| return (int)(wch - (WCHAR)'A' + 10); |
| if ( wch >= (WCHAR)'a' && wch <= (WCHAR)'f' ) |
| return (int)(wch - (WCHAR)'a' + 10); |
| return -1; |
| } |
| |
| static const WCHAR* get_hex(const WCHAR* pwsz,BYTE* pb) |
| { |
| int hi,lo; |
| |
| *pb = 0; |
| if ( pwsz == NULL ) return NULL; |
| hi = wchar_to_hex(*pwsz); if ( hi < 0 ) return NULL; pwsz++; |
| lo = wchar_to_hex(*pwsz); if ( lo < 0 ) return NULL; pwsz++; |
| *pb = (BYTE)( (hi << 4) | lo ); |
| return pwsz; |
| } |
| |
| static const WCHAR* skip_hex(const WCHAR* pwsz) |
| { |
| if ( pwsz == NULL ) return NULL; |
| while ( 1 ) |
| { |
| if ( wchar_to_hex(*pwsz) < 0 ) |
| break; |
| pwsz++; |
| } |
| return pwsz; |
| } |
| |
| static const WCHAR* next_token(const WCHAR* pwsz) |
| { |
| if ( pwsz == NULL ) return NULL; |
| pwsz = skip_space(pwsz); |
| if ( *pwsz != (WCHAR)',' ) return NULL; pwsz++; |
| return skip_space(pwsz); |
| } |
| |
| |
| static HRESULT QUARTZ_SourceTypeIsMatch( |
| const BYTE* pData, DWORD cbData, |
| const WCHAR* pwszTempl, DWORD cchTempl ) |
| { |
| DWORD dwOfs; |
| DWORD n; |
| DWORD cbLen; |
| const WCHAR* pwszMask; |
| const WCHAR* pwszValue; |
| BYTE bMask; |
| BYTE bValue; |
| |
| TRACE("(%p,%lu,%s,%lu)\n",pData,cbData,debugstr_w(pwszTempl),cchTempl); |
| |
| pwszTempl = skip_space(pwszTempl); |
| while ( 1 ) |
| { |
| pwszTempl = get_dword(pwszTempl,&dwOfs); |
| pwszTempl = next_token(pwszTempl); |
| pwszTempl = get_dword(pwszTempl,&cbLen); |
| pwszMask = pwszTempl = next_token(pwszTempl); |
| pwszTempl = skip_hex(pwszTempl); |
| pwszValue = pwszTempl = next_token(pwszTempl); |
| pwszTempl = skip_hex(pwszValue); |
| pwszTempl = skip_space(pwszTempl); |
| if ( pwszValue == NULL ) |
| { |
| WARN( "parse error\n" ); |
| return S_FALSE; |
| } |
| |
| if ( dwOfs >= cbData || ( (dwOfs+cbLen) >= cbData ) ) |
| { |
| WARN( "length of given data is too short\n" ); |
| return S_FALSE; |
| } |
| |
| for ( n = 0; n < cbLen; n++ ) |
| { |
| pwszMask = get_hex(pwszMask,&bMask); |
| if ( pwszMask == NULL ) bMask = 0xff; |
| pwszValue = get_hex(pwszValue,&bValue); |
| if ( pwszValue == NULL ) |
| { |
| WARN( "parse error - invalid hex data\n" ); |
| return S_FALSE; |
| } |
| if ( (pData[dwOfs+n]&bMask) != (bValue&bMask) ) |
| { |
| TRACE( "not matched\n" ); |
| return S_FALSE; |
| } |
| } |
| |
| if ( *pwszTempl == 0 ) |
| break; |
| pwszTempl = next_token(pwszTempl); |
| if ( pwszTempl == NULL ) |
| { |
| WARN( "parse error\n" ); |
| return S_FALSE; |
| } |
| } |
| |
| TRACE( "matched\n" ); |
| return NOERROR; |
| } |
| |
| static HRESULT QUARTZ_GetSourceTypeFromData( |
| const BYTE* pData, DWORD cbData, |
| GUID* pidMajor, GUID* pidSub, CLSID* pidSource ) |
| { |
| HRESULT hr = S_FALSE; |
| LONG lr; |
| WCHAR wszMajor[128]; |
| WCHAR wszSub[128]; |
| WCHAR wszSource[128]; |
| WCHAR wszSourceFilter[128]; |
| WCHAR* pwszLocalBuf = NULL; |
| WCHAR* pwszTemp; |
| DWORD cbLocalBuf = 0; |
| DWORD cbPath; |
| DWORD dwIndexMajor; |
| HKEY hkMajor; |
| DWORD dwIndexSub; |
| HKEY hkSub; |
| DWORD dwIndexSource; |
| HKEY hkSource; |
| DWORD dwRegType; |
| DWORD cbRegData; |
| FILETIME ftLastWrite; |
| static const WCHAR wszFmt[] = {'%','l','u',0}; |
| |
| if ( RegOpenKeyExW( HKEY_CLASSES_ROOT, QUARTZ_wszMediaType, 0, KEY_READ, &hkMajor ) == ERROR_SUCCESS ) |
| { |
| dwIndexMajor = 0; |
| while ( hr == S_FALSE ) |
| { |
| cbPath = NUMELEMS(wszMajor)-1; |
| lr = RegEnumKeyExW( |
| hkMajor, dwIndexMajor ++, wszMajor, &cbPath, |
| NULL, NULL, NULL, &ftLastWrite ); |
| if ( lr != ERROR_SUCCESS ) |
| break; |
| if ( RegOpenKeyExW( hkMajor, wszMajor, 0, KEY_READ, &hkSub ) == ERROR_SUCCESS ) |
| { |
| dwIndexSub = 0; |
| while ( hr == S_FALSE ) |
| { |
| cbPath = NUMELEMS(wszSub)-1; |
| lr = RegEnumKeyExW( |
| hkSub, dwIndexSub ++, wszSub, &cbPath, |
| NULL, NULL, NULL, &ftLastWrite ); |
| if ( lr != ERROR_SUCCESS ) |
| break; |
| if ( RegOpenKeyExW( hkSub, wszSub, 0, KEY_READ, &hkSource ) == ERROR_SUCCESS ) |
| { |
| dwIndexSource = 0; |
| while ( hr == S_FALSE ) |
| { |
| wsprintfW(wszSource,wszFmt,dwIndexSource++); |
| lr = RegQueryValueExW( |
| hkSource, wszSource, NULL, |
| &dwRegType, NULL, &cbRegData ); |
| if ( lr != ERROR_SUCCESS ) |
| break; |
| if ( cbLocalBuf < cbRegData ) |
| { |
| pwszTemp = (WCHAR*)QUARTZ_ReallocMem( pwszLocalBuf, cbRegData+sizeof(WCHAR) ); |
| if ( pwszTemp == NULL ) |
| { |
| hr = E_OUTOFMEMORY; |
| break; |
| } |
| pwszLocalBuf = pwszTemp; |
| cbLocalBuf = cbRegData+sizeof(WCHAR); |
| } |
| cbRegData = cbLocalBuf; |
| lr = RegQueryValueExW( |
| hkSource, wszSource, NULL, |
| &dwRegType, (BYTE*)pwszLocalBuf, &cbRegData ); |
| if ( lr != ERROR_SUCCESS ) |
| break; |
| |
| hr = QUARTZ_SourceTypeIsMatch( |
| pData, cbData, |
| pwszLocalBuf, cbRegData / sizeof(WCHAR) ); |
| if ( hr == S_OK ) |
| { |
| hr = CLSIDFromString(wszMajor,pidMajor); |
| if ( hr == NOERROR ) |
| hr = CLSIDFromString(wszSub,pidSub); |
| if ( hr == NOERROR ) |
| { |
| lstrcpyW(wszSource,QUARTZ_wszSourceFilter); |
| cbRegData = NUMELEMS(wszSourceFilter)-sizeof(WCHAR); |
| lr = RegQueryValueExW( |
| hkSource, wszSource, NULL, |
| &dwRegType, |
| (BYTE*)wszSourceFilter, &cbRegData ); |
| if ( lr == ERROR_SUCCESS ) |
| { |
| hr = CLSIDFromString(wszSourceFilter,pidSource); |
| } |
| else |
| hr = E_FAIL; |
| } |
| |
| if ( hr != NOERROR && SUCCEEDED(hr) ) |
| hr = E_FAIL; |
| } |
| if ( hr != S_FALSE ) |
| break; |
| } |
| RegCloseKey( hkSource ); |
| } |
| } |
| RegCloseKey( hkSub ); |
| } |
| } |
| RegCloseKey( hkMajor ); |
| } |
| |
| if ( pwszLocalBuf != NULL ) |
| QUARTZ_FreeMem(pwszLocalBuf); |
| |
| return hr; |
| } |
| |
| |
| |
| /*************************************************************************** |
| * |
| * CFilterGraph::IFilterGraph2 methods |
| * |
| */ |
| |
| static HRESULT WINAPI |
| IFilterGraph2_fnQueryInterface(IFilterGraph2* iface,REFIID riid,void** ppobj) |
| { |
| CFilterGraph_THIS(iface,fgraph); |
| |
| TRACE("(%p)->()\n",This); |
| |
| return IUnknown_QueryInterface(This->unk.punkControl,riid,ppobj); |
| } |
| |
| static ULONG WINAPI |
| IFilterGraph2_fnAddRef(IFilterGraph2* iface) |
| { |
| CFilterGraph_THIS(iface,fgraph); |
| |
| TRACE("(%p)->()\n",This); |
| |
| return IUnknown_AddRef(This->unk.punkControl); |
| } |
| |
| static ULONG WINAPI |
| IFilterGraph2_fnRelease(IFilterGraph2* iface) |
| { |
| CFilterGraph_THIS(iface,fgraph); |
| |
| TRACE("(%p)->()\n",This); |
| |
| return IUnknown_Release(This->unk.punkControl); |
| } |
| |
| static HRESULT WINAPI |
| IFilterGraph2_fnAddFilter(IFilterGraph2* iface,IBaseFilter* pFilter, LPCWSTR pName) |
| { |
| CFilterGraph_THIS(iface,fgraph); |
| FILTER_STATE fs; |
| FILTER_INFO info; |
| IBaseFilter* pTempFilter; |
| FG_FilterData* pActiveFiltersNew; |
| HRESULT hr; |
| HRESULT hrSucceeded = S_OK; |
| int i,iLen; |
| |
| TRACE( "(%p)->(%p,%s)\n",This,pFilter,debugstr_w(pName) ); |
| |
| EnterCriticalSection( &This->m_csFilters ); |
| |
| hr = IMediaFilter_GetState(CFilterGraph_IMediaFilter(This),0,&fs); |
| if ( hr == VFW_S_STATE_INTERMEDIATE ) |
| hr = VFW_E_STATE_CHANGED; |
| if ( fs != State_Stopped ) |
| hr = VFW_E_NOT_STOPPED; |
| if ( FAILED(hr) ) |
| goto end; |
| |
| TRACE( "(%p) search the specified name.\n",This ); |
| |
| if ( pName != NULL ) |
| { |
| hr = IFilterGraph2_FindFilterByName( CFilterGraph_IFilterGraph2(This), pName, &pTempFilter ); |
| if ( hr == S_OK ) |
| { |
| IBaseFilter_Release(pTempFilter); |
| hrSucceeded = VFW_S_DUPLICATE_NAME; |
| } |
| else |
| { |
| goto name_ok; |
| } |
| |
| iLen = lstrlenW(pName); |
| if ( iLen > 32 ) |
| iLen = 32; |
| memcpy( info.achName, pName, sizeof(WCHAR)*iLen ); |
| info.achName[iLen] = 0; |
| } |
| else |
| { |
| ZeroMemory( &info, sizeof(info) ); |
| hr = IBaseFilter_QueryFilterInfo( pFilter, &info ); |
| if ( FAILED(hr) ) |
| goto end; |
| if ( info.pGraph != NULL ) |
| { |
| IFilterGraph_Release(info.pGraph); |
| hr = E_FAIL; /* FIXME */ |
| goto end; |
| } |
| |
| hr = IFilterGraph2_FindFilterByName( CFilterGraph_IFilterGraph2(This), pName, &pTempFilter ); |
| if ( hr != S_OK ) |
| { |
| pName = info.achName; |
| goto name_ok; |
| } |
| } |
| |
| /* generate modified names for this filter.. */ |
| iLen = lstrlenW(info.achName); |
| if ( iLen > 32 ) |
| iLen = 32; |
| info.achName[iLen++] = ' '; |
| |
| for ( i = 0; i <= 99; i++ ) |
| { |
| info.achName[iLen+0] = (i%10) + '0'; |
| info.achName[iLen+1] = ((i/10)%10) + '0'; |
| info.achName[iLen+2] = 0; |
| |
| hr = IFilterGraph2_FindFilterByName( CFilterGraph_IFilterGraph2(This), info.achName, &pTempFilter ); |
| if ( hr != S_OK ) |
| { |
| pName = info.achName; |
| goto name_ok; |
| } |
| } |
| |
| hr = ( pName == NULL ) ? E_FAIL : VFW_E_DUPLICATE_NAME; |
| goto end; |
| |
| name_ok: |
| TRACE( "(%p) add this filter - %s.\n",This,debugstr_w(pName) ); |
| |
| /* register this filter. */ |
| pActiveFiltersNew = (FG_FilterData*)QUARTZ_ReallocMem( |
| This->m_pActiveFilters, |
| sizeof(FG_FilterData) * (This->m_cActiveFilters+1) ); |
| if ( pActiveFiltersNew == NULL ) |
| { |
| hr = E_OUTOFMEMORY; |
| goto end; |
| } |
| This->m_pActiveFilters = pActiveFiltersNew; |
| pActiveFiltersNew = &This->m_pActiveFilters[This->m_cActiveFilters]; |
| |
| pActiveFiltersNew->pFilter = NULL; |
| pActiveFiltersNew->pPosition = NULL; |
| pActiveFiltersNew->pSeeking = NULL; |
| pActiveFiltersNew->pwszName = NULL; |
| pActiveFiltersNew->cbName = 0; |
| |
| pActiveFiltersNew->cbName = sizeof(WCHAR)*(lstrlenW(pName)+1); |
| pActiveFiltersNew->pwszName = (WCHAR*)QUARTZ_AllocMem( pActiveFiltersNew->cbName ); |
| if ( pActiveFiltersNew->pwszName == NULL ) |
| { |
| hr = E_OUTOFMEMORY; |
| goto end; |
| } |
| memcpy( pActiveFiltersNew->pwszName, pName, pActiveFiltersNew->cbName ); |
| |
| pActiveFiltersNew->pFilter = pFilter; |
| IBaseFilter_AddRef(pFilter); |
| IBaseFilter_QueryInterface( pFilter, &IID_IMediaPosition, (void**)&pActiveFiltersNew->pPosition ); |
| IBaseFilter_QueryInterface( pFilter, &IID_IMediaSeeking, (void**)&pActiveFiltersNew->pSeeking ); |
| |
| This->m_cActiveFilters ++; |
| |
| hr = IBaseFilter_JoinFilterGraph(pFilter,(IFilterGraph*)iface,pName); |
| if ( SUCCEEDED(hr) ) |
| { |
| EnterCriticalSection( &This->m_csClock ); |
| hr = IBaseFilter_SetSyncSource( pFilter, This->m_pClock ); |
| LeaveCriticalSection( &This->m_csClock ); |
| } |
| if ( FAILED(hr) ) |
| { |
| IBaseFilter_JoinFilterGraph(pFilter,NULL,pName); |
| IFilterGraph2_RemoveFilter(CFilterGraph_IFilterGraph2(This),pFilter); |
| goto end; |
| } |
| |
| hr = CFilterGraph_GraphChanged(This); |
| if ( FAILED(hr) ) |
| goto end; |
| |
| hr = hrSucceeded; |
| end: |
| LeaveCriticalSection( &This->m_csFilters ); |
| |
| TRACE( "(%p) return %08lx\n", This, hr ); |
| |
| return hr; |
| } |
| |
| static HRESULT WINAPI |
| IFilterGraph2_fnRemoveFilter(IFilterGraph2* iface,IBaseFilter* pFilter) |
| { |
| CFilterGraph_THIS(iface,fgraph); |
| FILTER_STATE fs; |
| DWORD n,copy; |
| HRESULT hr = NOERROR; |
| |
| TRACE( "(%p)->(%p)\n",This,pFilter ); |
| |
| EnterCriticalSection( &This->m_csFilters ); |
| |
| hr = IMediaFilter_GetState(CFilterGraph_IMediaFilter(This),0,&fs); |
| if ( hr == VFW_S_STATE_INTERMEDIATE ) |
| hr = VFW_E_STATE_CHANGED; |
| if ( fs != State_Stopped ) |
| hr = VFW_E_NOT_STOPPED; |
| if ( FAILED(hr) ) |
| goto end; |
| |
| hr = S_FALSE; /* FIXME? */ |
| |
| for ( n = 0; n < This->m_cActiveFilters; n++ ) |
| { |
| if ( This->m_pActiveFilters[n].pFilter == pFilter ) |
| { |
| CFilterGraph_DisconnectAllPins(pFilter); |
| (void)IBaseFilter_SetSyncSource( pFilter, NULL ); |
| (void)IBaseFilter_JoinFilterGraph( |
| pFilter, NULL, This->m_pActiveFilters[n].pwszName ); |
| |
| if ( This->m_pActiveFilters[n].pFilter != NULL ) |
| IBaseFilter_Release(This->m_pActiveFilters[n].pFilter); |
| if ( This->m_pActiveFilters[n].pPosition != NULL ) |
| IMediaPosition_Release(This->m_pActiveFilters[n].pPosition); |
| if ( This->m_pActiveFilters[n].pSeeking != NULL ) |
| IMediaSeeking_Release(This->m_pActiveFilters[n].pSeeking); |
| if ( This->m_pActiveFilters[n].pwszName != NULL ) |
| QUARTZ_FreeMem(This->m_pActiveFilters[n].pwszName); |
| |
| copy = This->m_cActiveFilters - n - 1; |
| if ( copy > 0 ) |
| memmove( &This->m_pActiveFilters[n], |
| &This->m_pActiveFilters[n+1], |
| sizeof(FG_FilterData) * copy ); |
| This->m_cActiveFilters --; |
| |
| hr = CFilterGraph_GraphChanged(This); |
| break; |
| } |
| } |
| |
| end:; |
| LeaveCriticalSection( &This->m_csFilters ); |
| |
| return hr; |
| } |
| |
| static HRESULT WINAPI |
| IFilterGraph2_fnEnumFilters(IFilterGraph2* iface,IEnumFilters** ppEnum) |
| { |
| CFilterGraph_THIS(iface,fgraph); |
| QUARTZ_CompList* pList = NULL; |
| DWORD n; |
| HRESULT hr; |
| |
| TRACE( "(%p)->(%p)\n",This,ppEnum ); |
| |
| EnterCriticalSection( &This->m_csFilters ); |
| |
| pList = QUARTZ_CompList_Alloc(); |
| if ( pList == NULL ) |
| { |
| hr = E_OUTOFMEMORY; |
| goto err; |
| } |
| for ( n = 0; n < This->m_cActiveFilters; n++ ) |
| { |
| hr = QUARTZ_CompList_AddTailComp( |
| pList, (IUnknown*)This->m_pActiveFilters[n].pFilter, NULL, 0 ); |
| if ( FAILED(hr) ) |
| goto err; |
| } |
| |
| hr = QUARTZ_CreateEnumUnknown( |
| &IID_IEnumFilters, (void**)ppEnum, pList ); |
| err: |
| if ( pList != NULL ) |
| QUARTZ_CompList_Free( pList ); |
| |
| LeaveCriticalSection( &This->m_csFilters ); |
| |
| return hr; |
| } |
| |
| static HRESULT WINAPI |
| IFilterGraph2_fnFindFilterByName(IFilterGraph2* iface,LPCWSTR pName,IBaseFilter** ppFilter) |
| { |
| CFilterGraph_THIS(iface,fgraph); |
| DWORD n; |
| DWORD cbName; |
| HRESULT hr = E_FAIL; /* FIXME */ |
| |
| TRACE( "(%p)->(%s,%p)\n",This,debugstr_w(pName),ppFilter ); |
| |
| if ( pName == NULL || ppFilter == NULL ) |
| return E_POINTER; |
| *ppFilter = NULL; |
| |
| cbName = sizeof(WCHAR) * (lstrlenW(pName) + 1); |
| |
| EnterCriticalSection( &This->m_csFilters ); |
| |
| for ( n = 0; n < This->m_cActiveFilters; n++ ) |
| { |
| if ( This->m_pActiveFilters[n].cbName == cbName && |
| !memcmp( This->m_pActiveFilters[n].pwszName, pName, cbName ) ) |
| { |
| *ppFilter = This->m_pActiveFilters[n].pFilter; |
| IBaseFilter_AddRef(*ppFilter); |
| hr = NOERROR; |
| } |
| } |
| |
| LeaveCriticalSection( &This->m_csFilters ); |
| |
| return hr; |
| } |
| |
| static HRESULT WINAPI |
| IFilterGraph2_fnConnectDirect(IFilterGraph2* iface,IPin* pOut,IPin* pIn,const AM_MEDIA_TYPE* pmt) |
| { |
| CFilterGraph_THIS(iface,fgraph); |
| IPin* pConnTo; |
| PIN_INFO infoIn; |
| PIN_INFO infoOut; |
| FILTER_INFO finfoIn; |
| FILTER_INFO finfoOut; |
| FILTER_STATE fs; |
| HRESULT hr; |
| |
| TRACE( "(%p)->(%p,%p,%p)\n",This,pOut,pIn,pmt ); |
| |
| infoIn.pFilter = NULL; |
| infoOut.pFilter = NULL; |
| finfoIn.pGraph = NULL; |
| finfoOut.pGraph = NULL; |
| |
| EnterCriticalSection( &This->m_csFilters ); |
| |
| hr = IMediaFilter_GetState(CFilterGraph_IMediaFilter(This),0,&fs); |
| if ( hr == VFW_S_STATE_INTERMEDIATE ) |
| hr = VFW_E_STATE_CHANGED; |
| if ( fs != State_Stopped ) |
| hr = VFW_E_NOT_STOPPED; |
| if ( FAILED(hr) ) |
| goto end; |
| |
| hr = IPin_QueryPinInfo(pIn,&infoIn); |
| if ( FAILED(hr) ) |
| goto end; |
| hr = IPin_QueryPinInfo(pOut,&infoOut); |
| if ( FAILED(hr) ) |
| goto end; |
| if ( infoIn.pFilter == NULL || infoOut.pFilter == NULL || |
| infoIn.dir != PINDIR_INPUT || infoOut.dir != PINDIR_OUTPUT ) |
| { |
| hr = E_FAIL; |
| goto end; |
| } |
| |
| hr = IBaseFilter_QueryFilterInfo(infoIn.pFilter,&finfoIn); |
| if ( FAILED(hr) ) |
| goto end; |
| hr = IBaseFilter_QueryFilterInfo(infoOut.pFilter,&finfoOut); |
| if ( FAILED(hr) ) |
| goto end; |
| if ( finfoIn.pGraph != ((IFilterGraph*)iface) || |
| finfoOut.pGraph != ((IFilterGraph*)iface) ) |
| { |
| hr = E_FAIL; |
| goto end; |
| } |
| |
| pConnTo = NULL; |
| hr = IPin_ConnectedTo(pIn,&pConnTo); |
| if ( hr == NOERROR && pConnTo != NULL ) |
| { |
| IPin_Release(pConnTo); |
| hr = VFW_E_ALREADY_CONNECTED; |
| goto end; |
| } |
| |
| pConnTo = NULL; |
| hr = IPin_ConnectedTo(pOut,&pConnTo); |
| if ( hr == NOERROR && pConnTo != NULL ) |
| { |
| IPin_Release(pConnTo); |
| hr = VFW_E_ALREADY_CONNECTED; |
| goto end; |
| } |
| |
| TRACE("(%p) try to connect %p<->%p\n",This,pIn,pOut); |
| hr = IPin_Connect(pOut,pIn,pmt); |
| if ( FAILED(hr) ) |
| { |
| TRACE("(%p)->Connect(%p,%p) hr = %08lx\n",pOut,pIn,pmt,hr); |
| IPin_Disconnect(pOut); |
| IPin_Disconnect(pIn); |
| goto end; |
| } |
| |
| hr = CFilterGraph_GraphChanged(This); |
| if ( FAILED(hr) ) |
| goto end; |
| |
| end: |
| LeaveCriticalSection( &This->m_csFilters ); |
| |
| if ( infoIn.pFilter != NULL ) |
| IBaseFilter_Release(infoIn.pFilter); |
| if ( infoOut.pFilter != NULL ) |
| IBaseFilter_Release(infoOut.pFilter); |
| if ( finfoIn.pGraph != NULL ) |
| IFilterGraph_Release(finfoIn.pGraph); |
| if ( finfoOut.pGraph != NULL ) |
| IFilterGraph_Release(finfoOut.pGraph); |
| |
| return hr; |
| } |
| |
| static HRESULT WINAPI |
| IFilterGraph2_fnReconnect(IFilterGraph2* iface,IPin* pPin) |
| { |
| CFilterGraph_THIS(iface,fgraph); |
| |
| TRACE( "(%p)->(%p)\n",This,pPin ); |
| |
| return IFilterGraph2_ReconnectEx(iface,pPin,NULL); |
| } |
| |
| static HRESULT WINAPI |
| IFilterGraph2_fnDisconnect(IFilterGraph2* iface,IPin* pPin) |
| { |
| CFilterGraph_THIS(iface,fgraph); |
| IPin* pConnTo; |
| HRESULT hr; |
| |
| TRACE( "(%p)->(%p)\n",This,pPin ); |
| |
| EnterCriticalSection( &This->m_csFilters ); |
| |
| pConnTo = NULL; |
| hr = IPin_ConnectedTo(pPin,&pConnTo); |
| if ( hr == NOERROR && pConnTo != NULL ) |
| { |
| IPin_Disconnect(pConnTo); |
| IPin_Release(pConnTo); |
| } |
| hr = IPin_Disconnect(pPin); |
| if ( FAILED(hr) ) |
| goto end; |
| |
| hr = CFilterGraph_GraphChanged(This); |
| if ( FAILED(hr) ) |
| goto end; |
| |
| end: |
| LeaveCriticalSection( &This->m_csFilters ); |
| |
| return hr; |
| } |
| |
| static HRESULT WINAPI |
| IFilterGraph2_fnSetDefaultSyncSource(IFilterGraph2* iface) |
| { |
| CFilterGraph_THIS(iface,fgraph); |
| IUnknown* punk; |
| IReferenceClock* pClock; |
| HRESULT hr; |
| |
| FIXME( "(%p)->() stub!\n", This ); |
| |
| /* FIXME - search all filters from renderer. */ |
| |
| hr = QUARTZ_CreateSystemClock( NULL, (void**)&punk ); |
| if ( FAILED(hr) ) |
| return hr; |
| hr = IUnknown_QueryInterface( punk, &IID_IReferenceClock, (void**)&pClock ); IUnknown_Release( punk ); |
| if ( FAILED(hr) ) |
| return hr; |
| |
| hr = IMediaFilter_SetSyncSource( |
| CFilterGraph_IMediaFilter(This), pClock ); |
| IReferenceClock_Release( pClock ); |
| |
| return hr; |
| } |
| |
| static HRESULT WINAPI |
| IFilterGraph2_fnConnect(IFilterGraph2* iface,IPin* pOut,IPin* pIn) |
| { |
| CFilterGraph_THIS(iface,fgraph); |
| IFilterMapper2* pMap2 = NULL; |
| IEnumMoniker* pEnumMon = NULL; |
| IMoniker* pMon = NULL; |
| IBaseFilter* pFilter = NULL; |
| IEnumPins* pEnumPin = NULL; |
| IPin* pPinTry = NULL; |
| PIN_INFO info; |
| PIN_DIRECTION pindir; |
| ULONG cReturned; |
| BOOL bTryConnect; |
| BOOL bConnected = FALSE; |
| CLSID clsidOutFilter; |
| CLSID clsidInFilter; |
| CLSID clsid; |
| HRESULT hr; |
| |
| TRACE( "(%p)->(%p,%p)\n",This,pOut,pIn ); |
| |
| /* At first, try to connect directly. */ |
| hr = IFilterGraph_ConnectDirect(iface,pOut,pIn,NULL); |
| if ( hr == NOERROR ) |
| return NOERROR; |
| |
| /* FIXME - try to connect indirectly. */ |
| FIXME( "(%p)->(%p,%p) not fully implemented\n",This,pOut,pIn ); |
| |
| info.pFilter = NULL; |
| hr = IPin_QueryPinInfo(pOut,&info); |
| if ( FAILED(hr) ) |
| return hr; |
| if ( info.pFilter == NULL ) |
| return E_FAIL; |
| hr = IBaseFilter_GetClassID(info.pFilter,&clsidOutFilter); |
| IBaseFilter_Release(info.pFilter); |
| if ( FAILED(hr) ) |
| return hr; |
| |
| info.pFilter = NULL; |
| hr = IPin_QueryPinInfo(pIn,&info); |
| if ( FAILED(hr) ) |
| return hr; |
| if ( info.pFilter == NULL ) |
| return E_FAIL; |
| hr = IBaseFilter_GetClassID(info.pFilter,&clsidInFilter); |
| IBaseFilter_Release(info.pFilter); |
| if ( FAILED(hr) ) |
| return hr; |
| |
| /* FIXME - try to connect with unused filters. */ |
| /* FIXME - try to connect with cached filters. */ |
| /* FIXME - enumerate transform filters and try to connect */ |
| hr = CoCreateInstance( |
| &CLSID_FilterMapper2, NULL, CLSCTX_INPROC_SERVER, |
| &IID_IFilterMapper2, (void**)&pMap2 ); |
| if ( FAILED(hr) ) |
| return hr; |
| hr = IFilterMapper2_EnumMatchingFilters( |
| pMap2,&pEnumMon,0,FALSE,MERIT_DO_NOT_USE+1, |
| TRUE,0,NULL,NULL,NULL,FALSE, |
| TRUE,0,NULL,NULL,NULL); |
| IFilterMapper2_Release(pMap2); |
| if ( FAILED(hr) ) |
| return hr; |
| TRACE("try to connect indirectly\n"); |
| |
| if ( hr == S_OK ) |
| { |
| while ( !bConnected && hr == S_OK ) |
| { |
| hr = IEnumMoniker_Next(pEnumMon,1,&pMon,&cReturned); |
| if ( hr != S_OK ) |
| break; |
| hr = IMoniker_BindToObject(pMon,NULL,NULL,&IID_IBaseFilter,(void**)&pFilter ); |
| if ( hr == S_OK ) |
| { |
| hr = IBaseFilter_GetClassID(pFilter,&clsid); |
| if ( hr == S_OK && |
| ( IsEqualGUID(&clsidOutFilter,&clsid) || IsEqualGUID(&clsidInFilter,&clsid) ) ) |
| hr = S_FALSE; |
| else |
| hr = IFilterGraph2_AddFilter(iface,pFilter,NULL); |
| if ( hr == S_OK ) |
| { |
| bTryConnect = FALSE; |
| hr = IBaseFilter_EnumPins(pFilter,&pEnumPin); |
| if ( hr == S_OK ) |
| { |
| { |
| while ( !bTryConnect ) |
| { |
| hr = IEnumPins_Next(pEnumPin,1,&pPinTry,&cReturned); |
| if ( hr != S_OK ) |
| break; |
| hr = IPin_QueryDirection(pPinTry,&pindir); |
| if ( hr == S_OK && pindir == PINDIR_INPUT ) |
| { |
| /* try to connect directly. */ |
| hr = IFilterGraph2_ConnectDirect(iface,pOut,pPinTry,NULL); |
| if ( hr == S_OK ) |
| bTryConnect = TRUE; |
| hr = S_OK; |
| } |
| IPin_Release(pPinTry); pPinTry = NULL; |
| } |
| } |
| IEnumPins_Release(pEnumPin); pEnumPin = NULL; |
| } |
| TRACE("TryConnect %d\n",bTryConnect); |
| |
| if ( bTryConnect ) |
| { |
| hr = IBaseFilter_EnumPins(pFilter,&pEnumPin); |
| if ( hr == S_OK ) |
| { |
| while ( !bConnected ) |
| { |
| hr = IEnumPins_Next(pEnumPin,1,&pPinTry,&cReturned); |
| if ( hr != S_OK ) |
| break; |
| hr = IPin_QueryDirection(pPinTry,&pindir); |
| if ( hr == S_OK && pindir == PINDIR_OUTPUT ) |
| { |
| /* try to connect indirectly. */ |
| hr = IFilterGraph2_Connect(iface,pPinTry,pIn); |
| if ( hr == S_OK ) |
| bConnected = TRUE; |
| hr = S_OK; |
| } |
| IPin_Release(pPinTry); pPinTry = NULL; |
| } |
| IEnumPins_Release(pEnumPin); pEnumPin = NULL; |
| } |
| } |
| if ( !bConnected ) |
| hr = IFilterGraph2_RemoveFilter(iface,pFilter); |
| } |
| IBaseFilter_Release(pFilter); pFilter = NULL; |
| if ( SUCCEEDED(hr) ) |
| hr = S_OK; |
| } |
| else |
| { |
| hr = S_OK; |
| } |
| IMoniker_Release(pMon); pMon = NULL; |
| } |
| IEnumMoniker_Release(pEnumMon); pEnumMon = NULL; |
| } |
| |
| if ( SUCCEEDED(hr) && !bConnected ) |
| hr = VFW_E_CANNOT_CONNECT; |
| |
| return hr; |
| } |
| |
| static HRESULT WINAPI |
| IFilterGraph2_fnRender(IFilterGraph2* iface,IPin* pOut) |
| { |
| CFilterGraph_THIS(iface,fgraph); |
| |
| TRACE( "(%p)->(%p)\n",This,pOut); |
| |
| return IFilterGraph2_RenderEx( CFilterGraph_IFilterGraph2(This), pOut, 0, NULL ); |
| } |
| |
| static HRESULT WINAPI |
| IFilterGraph2_fnRenderFile(IFilterGraph2* iface,LPCWSTR lpFileName,LPCWSTR lpPlayList) |
| { |
| CFilterGraph_THIS(iface,fgraph); |
| HRESULT hr; |
| IBaseFilter* pFilter = NULL; |
| IEnumPins* pEnum = NULL; |
| IPin* pPin; |
| ULONG cFetched; |
| PIN_DIRECTION dir; |
| ULONG cTryToRender; |
| ULONG cActRender; |
| |
| TRACE( "(%p)->(%s,%s)\n",This, |
| debugstr_w(lpFileName),debugstr_w(lpPlayList) ); |
| |
| if ( lpPlayList != NULL ) |
| return E_INVALIDARG; |
| |
| pFilter = NULL; |
| hr = IFilterGraph2_AddSourceFilter(iface,lpFileName,NULL,&pFilter); |
| if ( FAILED(hr) ) |
| goto end; |
| if ( pFilter == NULL ) |
| { |
| hr = E_FAIL; |
| goto end; |
| } |
| TRACE("(%p) source filter %p\n",This,pFilter); |
| |
| pEnum = NULL; |
| hr = IBaseFilter_EnumPins( pFilter, &pEnum ); |
| if ( FAILED(hr) ) |
| goto end; |
| if ( pEnum == NULL ) |
| { |
| hr = E_FAIL; |
| goto end; |
| } |
| |
| cTryToRender = 0; |
| cActRender = 0; |
| |
| while ( 1 ) |
| { |
| pPin = NULL; |
| cFetched = 0; |
| hr = IEnumPins_Next( pEnum, 1, &pPin, &cFetched ); |
| if ( FAILED(hr) ) |
| goto end; |
| if ( hr != NOERROR || pPin == NULL || cFetched != 1 ) |
| { |
| hr = NOERROR; |
| break; |
| } |
| hr = IPin_QueryDirection( pPin, &dir ); |
| if ( hr == NOERROR && dir == PINDIR_OUTPUT ) |
| { |
| cTryToRender ++; |
| hr = IFilterGraph2_Render( iface, pPin ); |
| if ( hr == NOERROR ) |
| cActRender ++; |
| } |
| IPin_Release( pPin ); |
| } |
| |
| if ( hr == NOERROR ) |
| { |
| if ( cTryToRender > cActRender ) |
| hr = VFW_S_PARTIAL_RENDER; |
| if ( cActRender == 0 ) |
| hr = E_FAIL; |
| } |
| |
| end: |
| if ( pEnum != NULL ) |
| IEnumPins_Release( pEnum ); |
| if ( pFilter != NULL ) |
| IBaseFilter_Release( pFilter ); |
| |
| return hr; |
| } |
| |
| static HRESULT WINAPI |
| IFilterGraph2_fnAddSourceFilter(IFilterGraph2* iface,LPCWSTR lpFileName,LPCWSTR lpFilterName,IBaseFilter** ppBaseFilter) |
| { |
| CFilterGraph_THIS(iface,fgraph); |
| HRESULT hr; |
| BYTE bStartData[512]; |
| DWORD cbStartData; |
| AM_MEDIA_TYPE mt; |
| CLSID clsidSource; |
| IFileSourceFilter* pSource; |
| |
| TRACE( "(%p)->(%s,%s,%p)\n",This, |
| debugstr_w(lpFileName),debugstr_w(lpFilterName),ppBaseFilter ); |
| |
| if ( lpFileName == NULL || ppBaseFilter == NULL ) |
| return E_POINTER; |
| *ppBaseFilter = NULL; |
| |
| hr = QUARTZ_PeekFile( lpFileName, bStartData, 512, &cbStartData ); |
| if ( FAILED(hr) ) |
| { |
| FIXME("cannot open %s (NOTE: URL is not implemented)\n", debugstr_w(lpFileName)); |
| return hr; |
| } |
| ZeroMemory( &mt, sizeof(AM_MEDIA_TYPE) ); |
| mt.bFixedSizeSamples = 1; |
| mt.lSampleSize = 1; |
| memcpy( &mt.majortype, &MEDIATYPE_Stream, sizeof(GUID) ); |
| memcpy( &mt.subtype, &MEDIASUBTYPE_NULL, sizeof(GUID) ); |
| memcpy( &mt.formattype, &FORMAT_None, sizeof(GUID) ); |
| hr = QUARTZ_GetSourceTypeFromData( |
| bStartData, cbStartData, |
| &mt.majortype, &mt.subtype, &clsidSource ); |
| if ( FAILED(hr) ) |
| { |
| ERR("QUARTZ_GetSourceTypeFromData() failed - return %08lx\n",hr); |
| return hr; |
| } |
| if ( hr != S_OK ) |
| { |
| FIXME( "file %s - unknown format\n", debugstr_w(lpFileName) ); |
| return VFW_E_INVALID_FILE_FORMAT; |
| } |
| |
| hr = CoCreateInstance( |
| &clsidSource, NULL, CLSCTX_INPROC_SERVER, |
| &IID_IBaseFilter, (void**)ppBaseFilter ); |
| if ( FAILED(hr) ) |
| return hr; |
| hr = IBaseFilter_QueryInterface(*ppBaseFilter,&IID_IFileSourceFilter,(void**)&pSource); |
| if ( SUCCEEDED(hr) ) |
| { |
| hr = IFileSourceFilter_Load(pSource,lpFileName,&mt); |
| IFileSourceFilter_Release(pSource); |
| } |
| if ( SUCCEEDED(hr) ) |
| hr = IFilterGraph2_AddFilter(iface,*ppBaseFilter,lpFilterName); |
| if ( FAILED(hr) ) |
| { |
| IBaseFilter_Release(*ppBaseFilter); |
| *ppBaseFilter = NULL; |
| return hr; |
| } |
| |
| return S_OK; |
| } |
| |
| static HRESULT WINAPI |
| IFilterGraph2_fnSetLogFile(IFilterGraph2* iface,DWORD_PTR hFile) |
| { |
| CFilterGraph_THIS(iface,fgraph); |
| |
| FIXME( "(%p)->() stub!\n", This ); |
| |
| return S_OK; /* no debug output */ |
| } |
| |
| static HRESULT WINAPI |
| IFilterGraph2_fnAbort(IFilterGraph2* iface) |
| { |
| CFilterGraph_THIS(iface,fgraph); |
| |
| FIXME( "(%p)->() stub!\n", This ); |
| |
| /* FIXME - abort the current asynchronous task. */ |
| |
| return S_OK; |
| } |
| |
| static HRESULT WINAPI |
| IFilterGraph2_fnShouldOperationContinue(IFilterGraph2* iface) |
| { |
| CFilterGraph_THIS(iface,fgraph); |
| |
| FIXME( "(%p)->() stub!\n", This ); |
| |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI |
| IFilterGraph2_fnAddSourceFilterForMoniker(IFilterGraph2* iface,IMoniker* pMon,IBindCtx* pCtx,LPCWSTR pFilterName,IBaseFilter** ppFilter) |
| { |
| CFilterGraph_THIS(iface,fgraph); |
| |
| FIXME( "(%p)->() stub!\n", This ); |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI |
| IFilterGraph2_fnReconnectEx(IFilterGraph2* iface,IPin* pPin,const AM_MEDIA_TYPE* pmt) |
| { |
| CFilterGraph_THIS(iface,fgraph); |
| HRESULT hr; |
| |
| FIXME( "(%p)->(%p,%p) stub!\n",This,pPin,pmt ); |
| |
| /* reconnect asynchronously. */ |
| |
| EnterCriticalSection( &This->m_csFilters ); |
| hr = CFilterGraph_GraphChanged(This); |
| LeaveCriticalSection( &This->m_csFilters ); |
| |
| return E_NOTIMPL; |
| } |
| |
| static HRESULT WINAPI |
| IFilterGraph2_fnRenderEx(IFilterGraph2* iface,IPin* pOut,DWORD dwFlags,DWORD* pdwReserved) |
| { |
| CFilterGraph_THIS(iface,fgraph); |
| HRESULT hr; |
| IFilterMapper2* pMap2 = NULL; |
| IEnumMoniker* pEnumMon = NULL; |
| IMoniker* pMon = NULL; |
| IBaseFilter* pFilter = NULL; |
| IEnumPins* pEnumPin = NULL; |
| IPin* pPin = NULL; |
| PIN_DIRECTION pindir; |
| BOOL bRendered = FALSE; |
| ULONG cReturned; |
| |
| TRACE( "(%p)->(%p,%08lx,%p)\n",This,pPin,dwFlags,pdwReserved); |
| |
| if ( pdwReserved != NULL ) |
| return E_INVALIDARG; |
| |
| if ( dwFlags != 0 ) |
| { |
| FIXME( "dwFlags != 0 (0x%08lx)\n", dwFlags ); |
| return E_INVALIDARG; |
| } |
| |
| /* FIXME - must be locked */ |
| /*EnterCriticalSection( &This->m_csFilters );*/ |
| |
| if ( pOut == NULL ) |
| return E_POINTER; |
| |
| hr = CoCreateInstance( |
| &CLSID_FilterMapper2, NULL, CLSCTX_INPROC_SERVER, |
| &IID_IFilterMapper2, (void**)&pMap2 ); |
| if ( FAILED(hr) ) |
| return hr; |
| hr = IFilterMapper2_EnumMatchingFilters( |
| pMap2,&pEnumMon,0,FALSE,MERIT_DO_NOT_USE+1, |
| TRUE,0,NULL,NULL,NULL,TRUE, |
| FALSE,0,NULL,NULL,NULL); |
| IFilterMapper2_Release(pMap2); |
| if ( FAILED(hr) ) |
| return hr; |
| TRACE("try to render pin\n"); |
| |
| if ( hr == S_OK ) |
| { |
| /* try to render pin. */ |
| while ( !bRendered && hr == S_OK ) |
| { |
| hr = IEnumMoniker_Next(pEnumMon,1,&pMon,&cReturned); |
| if ( hr != S_OK ) |
| break; |
| hr = IMoniker_BindToObject(pMon,NULL,NULL,&IID_IBaseFilter,(void**)&pFilter ); |
| if ( hr == S_OK ) |
| { |
| hr = IFilterGraph2_AddFilter(iface,pFilter,NULL); |
| if ( hr == S_OK ) |
| { |
| hr = IBaseFilter_EnumPins(pFilter,&pEnumPin); |
| if ( hr == S_OK ) |
| { |
| while ( !bRendered ) |
| { |
| hr = IEnumPins_Next(pEnumPin,1,&pPin,&cReturned); |
| if ( hr != S_OK ) |
| break; |
| hr = IPin_QueryDirection(pPin,&pindir); |
| if ( hr == S_OK && pindir == PINDIR_INPUT ) |
| { |
| /* try to connect. */ |
| hr = IFilterGraph2_Connect(iface,pOut,pPin); |
| if ( hr == S_OK ) |
| bRendered = TRUE; |
| hr = S_OK; |
| } |
| IPin_Release(pPin); pPin = NULL; |
| } |
| IEnumPins_Release(pEnumPin); pEnumPin = NULL; |
| } |
| if ( !bRendered ) |
| hr = IFilterGraph2_RemoveFilter(iface,pFilter); |
| } |
| IBaseFilter_Release(pFilter); pFilter = NULL; |
| if ( SUCCEEDED(hr) ) |
| hr = S_OK; |
| } |
| else |
| { |
| hr = S_OK; |
| } |
| IMoniker_Release(pMon); pMon = NULL; |
| } |
| IEnumMoniker_Release(pEnumMon); pEnumMon = NULL; |
| } |
| |
| if ( bRendered ) |
| { |
| /* successfully rendered(but may be partial now) */ |
| hr = S_OK; |
| |
| /* FIXME - try to render all inserted filters. */ |
| /* hr = VFW_S_PARTIAL_RENDER; */ |
| } |
| else |
| { |
| if ( SUCCEEDED(hr) ) |
| hr = VFW_E_CANNOT_RENDER; |
| } |
| |
| /*LeaveCriticalSection( &This->m_csFilters );*/ |
| |
| return hr; |
| } |
| |
| |
| |
| |
| static ICOM_VTABLE(IFilterGraph2) ifgraph = |
| { |
| ICOM_MSVTABLE_COMPAT_DummyRTTIVALUE |
| /* IUnknown fields */ |
| IFilterGraph2_fnQueryInterface, |
| IFilterGraph2_fnAddRef, |
| IFilterGraph2_fnRelease, |
| /* IFilterGraph fields */ |
| IFilterGraph2_fnAddFilter, |
| IFilterGraph2_fnRemoveFilter, |
| IFilterGraph2_fnEnumFilters, |
| IFilterGraph2_fnFindFilterByName, |
| IFilterGraph2_fnConnectDirect, |
| IFilterGraph2_fnReconnect, |
| IFilterGraph2_fnDisconnect, |
| IFilterGraph2_fnSetDefaultSyncSource, |
| /* IGraphBuilder fields */ |
| IFilterGraph2_fnConnect, |
| IFilterGraph2_fnRender, |
| IFilterGraph2_fnRenderFile, |
| IFilterGraph2_fnAddSourceFilter, |
| IFilterGraph2_fnSetLogFile, |
| IFilterGraph2_fnAbort, |
| IFilterGraph2_fnShouldOperationContinue, |
| /* IFilterGraph2 fields */ |
| IFilterGraph2_fnAddSourceFilterForMoniker, |
| IFilterGraph2_fnReconnectEx, |
| IFilterGraph2_fnRenderEx, |
| }; |
| |
| HRESULT CFilterGraph_InitIFilterGraph2( CFilterGraph* pfg ) |
| { |
| TRACE("(%p)\n",pfg); |
| ICOM_VTBL(&pfg->fgraph) = &ifgraph; |
| |
| InitializeCriticalSection( &pfg->m_csFilters ); |
| pfg->m_cActiveFilters = 0; |
| pfg->m_pActiveFilters = NULL; |
| |
| return NOERROR; |
| } |
| |
| void CFilterGraph_UninitIFilterGraph2( CFilterGraph* pfg ) |
| { |
| TRACE("(%p)\n",pfg); |
| |
| /* remove all filters... */ |
| while ( pfg->m_cActiveFilters > 0 ) |
| { |
| IFilterGraph2_RemoveFilter( |
| CFilterGraph_IFilterGraph2(pfg), |
| pfg->m_pActiveFilters[pfg->m_cActiveFilters-1].pFilter ); |
| } |
| |
| if ( pfg->m_pActiveFilters != NULL ) |
| QUARTZ_FreeMem( pfg->m_pActiveFilters ); |
| |
| DeleteCriticalSection( &pfg->m_csFilters ); |
| } |
| |
| /*************************************************************************** |
| * |
| * CFilterGraph::IGraphVersion methods |
| * |
| */ |
| |
| static HRESULT WINAPI |
| IGraphVersion_fnQueryInterface(IGraphVersion* iface,REFIID riid,void** ppobj) |
| { |
| CFilterGraph_THIS(iface,graphversion); |
| |
| TRACE("(%p)->()\n",This); |
| |
| return IUnknown_QueryInterface(This->unk.punkControl,riid,ppobj); |
| } |
| |
| static ULONG WINAPI |
| IGraphVersion_fnAddRef(IGraphVersion* iface) |
| { |
| CFilterGraph_THIS(iface,graphversion); |
| |
| TRACE("(%p)->()\n",This); |
| |
| return IUnknown_AddRef(This->unk.punkControl); |
| } |
| |
| static ULONG WINAPI |
| IGraphVersion_fnRelease(IGraphVersion* iface) |
| { |
| CFilterGraph_THIS(iface,graphversion); |
| |
| TRACE("(%p)->()\n",This); |
| |
| return IUnknown_Release(This->unk.punkControl); |
| } |
| |
| |
| static HRESULT WINAPI |
| IGraphVersion_fnQueryVersion(IGraphVersion* iface,LONG* plVersion) |
| { |
| CFilterGraph_THIS(iface,graphversion); |
| |
| TRACE("(%p)->(%p)\n",This,plVersion); |
| |
| if ( plVersion == NULL ) |
| return E_POINTER; |
| |
| EnterCriticalSection( &This->m_csFilters ); |
| *plVersion = This->m_lGraphVersion; |
| LeaveCriticalSection( &This->m_csFilters ); |
| |
| return NOERROR; |
| } |
| |
| |
| static ICOM_VTABLE(IGraphVersion) igraphversion = |
| { |
| ICOM_MSVTABLE_COMPAT_DummyRTTIVALUE |
| /* IUnknown fields */ |
| IGraphVersion_fnQueryInterface, |
| IGraphVersion_fnAddRef, |
| IGraphVersion_fnRelease, |
| /* IGraphVersion fields */ |
| IGraphVersion_fnQueryVersion, |
| }; |
| |
| |
| |
| HRESULT CFilterGraph_InitIGraphVersion( CFilterGraph* pfg ) |
| { |
| TRACE("(%p)\n",pfg); |
| ICOM_VTBL(&pfg->graphversion) = &igraphversion; |
| |
| pfg->m_lGraphVersion = 1; |
| |
| return NOERROR; |
| } |
| |
| void CFilterGraph_UninitIGraphVersion( CFilterGraph* pfg ) |
| { |
| TRACE("(%p)\n",pfg); |
| } |
| |
| |