| /* |
| * Generic Implementation of IPin Interface |
| * |
| * Copyright 2003 Robert Shearman |
| * |
| * 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 "quartz_private.h" |
| #include "pin.h" |
| |
| #include "wine/debug.h" |
| #include "wine/unicode.h" |
| #include "uuids.h" |
| #include "vfwmsgs.h" |
| #include <assert.h> |
| |
| WINE_DEFAULT_DEBUG_CHANNEL(quartz); |
| |
| #define ALIGNDOWN(value,boundary) ((value)/(boundary)*(boundary)) |
| #define ALIGNUP(value,boundary) (ALIGNDOWN((value)+(boundary)-1, (boundary))) |
| |
| typedef HRESULT (*SendPinFunc)( IPin *to, LPVOID arg ); |
| |
| /** Helper function, there are a lot of places where the error code is inherited |
| * The following rules apply: |
| * |
| * Return the first received error code (E_NOTIMPL is ignored) |
| * If no errors occur: return the first received non-error-code that isn't S_OK |
| */ |
| static HRESULT updatehres( HRESULT original, HRESULT new ) |
| { |
| if (FAILED( original ) || new == E_NOTIMPL) |
| return original; |
| |
| if (FAILED( new ) || original == S_OK) |
| return new; |
| |
| return original; |
| } |
| |
| /** Sends a message from a pin further to other, similar pins |
| * fnMiddle is called on each pin found further on the stream. |
| * fnEnd (can be NULL) is called when the message can't be sent any further (this is a renderer or source) |
| * |
| * If the pin given is an input pin, the message will be sent downstream to other input pins |
| * If the pin given is an output pin, the message will be sent upstream to other output pins |
| */ |
| static HRESULT SendFurther( IPin *from, SendPinFunc fnMiddle, LPVOID arg, SendPinFunc fnEnd ) |
| { |
| PIN_INFO pin_info; |
| ULONG amount = 0; |
| HRESULT hr = S_OK; |
| HRESULT hr_return = S_OK; |
| IEnumPins *enumpins = NULL; |
| BOOL foundend = TRUE; |
| PIN_DIRECTION from_dir; |
| |
| IPin_QueryDirection( from, &from_dir ); |
| |
| hr = IPin_QueryInternalConnections( from, NULL, &amount ); |
| if (hr != E_NOTIMPL && amount) |
| FIXME("Use QueryInternalConnections!\n"); |
| |
| pin_info.pFilter = NULL; |
| hr = IPin_QueryPinInfo( from, &pin_info ); |
| if (FAILED(hr)) |
| goto out; |
| |
| hr = IBaseFilter_EnumPins( pin_info.pFilter, &enumpins ); |
| if (FAILED(hr)) |
| goto out; |
| |
| hr = IEnumPins_Reset( enumpins ); |
| while (hr == S_OK) { |
| IPin *pin = NULL; |
| hr = IEnumPins_Next( enumpins, 1, &pin, NULL ); |
| if (hr == VFW_E_ENUM_OUT_OF_SYNC) |
| { |
| hr = IEnumPins_Reset( enumpins ); |
| continue; |
| } |
| if (pin) |
| { |
| PIN_DIRECTION dir; |
| |
| IPin_QueryDirection( pin, &dir ); |
| if (dir != from_dir) |
| { |
| IPin *connected = NULL; |
| |
| foundend = FALSE; |
| IPin_ConnectedTo( pin, &connected ); |
| if (connected) |
| { |
| HRESULT hr_local; |
| |
| hr_local = fnMiddle( connected, arg ); |
| hr_return = updatehres( hr_return, hr_local ); |
| IPin_Release(connected); |
| } |
| } |
| IPin_Release( pin ); |
| } |
| else |
| { |
| hr = S_OK; |
| break; |
| } |
| } |
| |
| if (!foundend) |
| hr = hr_return; |
| else if (fnEnd) { |
| HRESULT hr_local; |
| |
| hr_local = fnEnd( from, arg ); |
| hr_return = updatehres( hr_return, hr_local ); |
| } |
| |
| out: |
| if (enumpins) |
| IEnumPins_Release( enumpins ); |
| if (pin_info.pFilter) |
| IBaseFilter_Release( pin_info.pFilter ); |
| return hr; |
| } |
| |
| |
| static void Copy_PinInfo(PIN_INFO * pDest, const PIN_INFO * pSrc) |
| { |
| /* Tempting to just do a memcpy, but the name field is |
| 128 characters long! We will probably never exceed 10 |
| most of the time, so we are better off copying |
| each field manually */ |
| strcpyW(pDest->achName, pSrc->achName); |
| pDest->dir = pSrc->dir; |
| pDest->pFilter = pSrc->pFilter; |
| } |
| |
| static HRESULT deliver_endofstream(IPin* pin, LPVOID unused) |
| { |
| return IPin_EndOfStream( pin ); |
| } |
| |
| static HRESULT deliver_beginflush(IPin* pin, LPVOID unused) |
| { |
| return IPin_BeginFlush( pin ); |
| } |
| |
| static HRESULT deliver_endflush(IPin* pin, LPVOID unused) |
| { |
| return IPin_EndFlush( pin ); |
| } |
| |
| typedef struct newsegmentargs |
| { |
| REFERENCE_TIME tStart, tStop; |
| double rate; |
| } newsegmentargs; |
| |
| static HRESULT deliver_newsegment(IPin *pin, LPVOID data) |
| { |
| newsegmentargs *args = data; |
| return IPin_NewSegment(pin, args->tStart, args->tStop, args->rate); |
| } |
| |
| /*** PullPin implementation ***/ |
| |
| static HRESULT PullPin_Init(const IPinVtbl *PullPin_Vtbl, const PIN_INFO * pPinInfo, SAMPLEPROC_PULL pSampleProc, LPVOID pUserData, |
| QUERYACCEPTPROC pQueryAccept, CLEANUPPROC pCleanUp, REQUESTPROC pCustomRequest, STOPPROCESSPROC pDone, LPCRITICAL_SECTION pCritSec, PullPin * pPinImpl) |
| { |
| /* Common attributes */ |
| pPinImpl->pin.IPin_iface.lpVtbl = PullPin_Vtbl; |
| pPinImpl->pin.refCount = 1; |
| pPinImpl->pin.pConnectedTo = NULL; |
| pPinImpl->pin.pCritSec = pCritSec; |
| Copy_PinInfo(&pPinImpl->pin.pinInfo, pPinInfo); |
| ZeroMemory(&pPinImpl->pin.mtCurrent, sizeof(AM_MEDIA_TYPE)); |
| |
| /* Input pin attributes */ |
| pPinImpl->pUserData = pUserData; |
| pPinImpl->fnQueryAccept = pQueryAccept; |
| pPinImpl->fnSampleProc = pSampleProc; |
| pPinImpl->fnCleanProc = pCleanUp; |
| pPinImpl->fnDone = pDone; |
| pPinImpl->fnPreConnect = NULL; |
| pPinImpl->pAlloc = NULL; |
| pPinImpl->prefAlloc = NULL; |
| pPinImpl->pReader = NULL; |
| pPinImpl->hThread = NULL; |
| pPinImpl->hEventStateChanged = CreateEventW(NULL, TRUE, TRUE, NULL); |
| pPinImpl->thread_sleepy = CreateEventW(NULL, FALSE, FALSE, NULL); |
| |
| pPinImpl->rtStart = 0; |
| pPinImpl->rtCurrent = 0; |
| pPinImpl->rtStop = ((LONGLONG)0x7fffffff << 32) | 0xffffffff; |
| pPinImpl->dRate = 1.0; |
| pPinImpl->state = Req_Die; |
| pPinImpl->fnCustomRequest = pCustomRequest; |
| pPinImpl->stop_playback = TRUE; |
| |
| InitializeCriticalSection(&pPinImpl->thread_lock); |
| pPinImpl->thread_lock.DebugInfo->Spare[0] = (DWORD_PTR)( __FILE__ ": PullPin.thread_lock"); |
| |
| return S_OK; |
| } |
| |
| HRESULT PullPin_Construct(const IPinVtbl *PullPin_Vtbl, const PIN_INFO * pPinInfo, SAMPLEPROC_PULL pSampleProc, LPVOID pUserData, QUERYACCEPTPROC pQueryAccept, CLEANUPPROC pCleanUp, REQUESTPROC pCustomRequest, STOPPROCESSPROC pDone, LPCRITICAL_SECTION pCritSec, IPin ** ppPin) |
| { |
| PullPin * pPinImpl; |
| |
| *ppPin = NULL; |
| |
| if (pPinInfo->dir != PINDIR_INPUT) |
| { |
| ERR("Pin direction(%x) != PINDIR_INPUT\n", pPinInfo->dir); |
| return E_INVALIDARG; |
| } |
| |
| pPinImpl = CoTaskMemAlloc(sizeof(*pPinImpl)); |
| |
| if (!pPinImpl) |
| return E_OUTOFMEMORY; |
| |
| if (SUCCEEDED(PullPin_Init(PullPin_Vtbl, pPinInfo, pSampleProc, pUserData, pQueryAccept, pCleanUp, pCustomRequest, pDone, pCritSec, pPinImpl))) |
| { |
| *ppPin = &pPinImpl->pin.IPin_iface; |
| return S_OK; |
| } |
| |
| CoTaskMemFree(pPinImpl); |
| return E_FAIL; |
| } |
| |
| static HRESULT PullPin_InitProcessing(PullPin * This); |
| |
| HRESULT WINAPI PullPin_ReceiveConnection(IPin * iface, IPin * pReceivePin, const AM_MEDIA_TYPE * pmt) |
| { |
| PIN_DIRECTION pindirReceive; |
| HRESULT hr = S_OK; |
| PullPin *This = impl_PullPin_from_IPin(iface); |
| |
| TRACE("(%p/%p)->(%p, %p)\n", This, iface, pReceivePin, pmt); |
| dump_AM_MEDIA_TYPE(pmt); |
| |
| EnterCriticalSection(This->pin.pCritSec); |
| if (!This->pin.pConnectedTo) |
| { |
| ALLOCATOR_PROPERTIES props; |
| |
| props.cBuffers = 3; |
| props.cbBuffer = 64 * 1024; /* 64 KB */ |
| props.cbAlign = 1; |
| props.cbPrefix = 0; |
| |
| if (This->fnQueryAccept(This->pUserData, pmt) != S_OK) |
| hr = VFW_E_TYPE_NOT_ACCEPTED; /* FIXME: shouldn't we just map common errors onto |
| * VFW_E_TYPE_NOT_ACCEPTED and pass the value on otherwise? */ |
| |
| if (SUCCEEDED(hr)) |
| { |
| IPin_QueryDirection(pReceivePin, &pindirReceive); |
| |
| if (pindirReceive != PINDIR_OUTPUT) |
| { |
| ERR("Can't connect from non-output pin\n"); |
| hr = VFW_E_INVALID_DIRECTION; |
| } |
| } |
| |
| This->pReader = NULL; |
| This->pAlloc = NULL; |
| This->prefAlloc = NULL; |
| if (SUCCEEDED(hr)) |
| { |
| hr = IPin_QueryInterface(pReceivePin, &IID_IAsyncReader, (LPVOID *)&This->pReader); |
| } |
| |
| if (SUCCEEDED(hr) && This->fnPreConnect) |
| { |
| hr = This->fnPreConnect(iface, pReceivePin, &props); |
| } |
| |
| /* |
| * Some custom filters (such as the one used by Fallout 3 |
| * and Fallout: New Vegas) expect to be passed a non-NULL |
| * preferred allocator. |
| */ |
| if (SUCCEEDED(hr)) |
| { |
| hr = StdMemAllocator_create(NULL, (LPVOID *) &This->prefAlloc); |
| } |
| |
| if (SUCCEEDED(hr)) |
| { |
| hr = IAsyncReader_RequestAllocator(This->pReader, This->prefAlloc, &props, &This->pAlloc); |
| } |
| |
| if (SUCCEEDED(hr)) |
| { |
| CopyMediaType(&This->pin.mtCurrent, pmt); |
| This->pin.pConnectedTo = pReceivePin; |
| IPin_AddRef(pReceivePin); |
| hr = IMemAllocator_Commit(This->pAlloc); |
| } |
| |
| if (SUCCEEDED(hr)) |
| hr = PullPin_InitProcessing(This); |
| |
| if (FAILED(hr)) |
| { |
| if (This->pReader) |
| IAsyncReader_Release(This->pReader); |
| This->pReader = NULL; |
| if (This->prefAlloc) |
| IMemAllocator_Release(This->prefAlloc); |
| This->prefAlloc = NULL; |
| if (This->pAlloc) |
| IMemAllocator_Release(This->pAlloc); |
| This->pAlloc = NULL; |
| } |
| } |
| else |
| hr = VFW_E_ALREADY_CONNECTED; |
| LeaveCriticalSection(This->pin.pCritSec); |
| return hr; |
| } |
| |
| HRESULT WINAPI PullPin_QueryInterface(IPin * iface, REFIID riid, LPVOID * ppv) |
| { |
| PullPin *This = impl_PullPin_from_IPin(iface); |
| |
| TRACE("(%p/%p)->(%s, %p)\n", This, iface, qzdebugstr_guid(riid), ppv); |
| |
| *ppv = NULL; |
| |
| if (IsEqualIID(riid, &IID_IUnknown)) |
| *ppv = iface; |
| else if (IsEqualIID(riid, &IID_IPin)) |
| *ppv = iface; |
| else if (IsEqualIID(riid, &IID_IMediaSeeking) || |
| IsEqualIID(riid, &IID_IQualityControl)) |
| { |
| return IBaseFilter_QueryInterface(This->pin.pinInfo.pFilter, riid, ppv); |
| } |
| |
| if (*ppv) |
| { |
| IUnknown_AddRef((IUnknown *)(*ppv)); |
| return S_OK; |
| } |
| |
| FIXME("No interface for %s!\n", qzdebugstr_guid(riid)); |
| |
| return E_NOINTERFACE; |
| } |
| |
| ULONG WINAPI PullPin_Release(IPin *iface) |
| { |
| PullPin *This = impl_PullPin_from_IPin(iface); |
| ULONG refCount = InterlockedDecrement(&This->pin.refCount); |
| |
| TRACE("(%p)->() Release from %d\n", This, refCount + 1); |
| |
| if (!refCount) |
| { |
| WaitForSingleObject(This->hEventStateChanged, INFINITE); |
| assert(!This->hThread); |
| |
| if(This->prefAlloc) |
| IMemAllocator_Release(This->prefAlloc); |
| if(This->pAlloc) |
| IMemAllocator_Release(This->pAlloc); |
| if(This->pReader) |
| IAsyncReader_Release(This->pReader); |
| CloseHandle(This->thread_sleepy); |
| CloseHandle(This->hEventStateChanged); |
| This->thread_lock.DebugInfo->Spare[0] = 0; |
| DeleteCriticalSection(&This->thread_lock); |
| CoTaskMemFree(This); |
| return 0; |
| } |
| return refCount; |
| } |
| |
| static void PullPin_Flush(PullPin *This) |
| { |
| IMediaSample *pSample; |
| TRACE("Flushing!\n"); |
| |
| if (This->pReader) |
| { |
| /* Do not allow state to change while flushing */ |
| EnterCriticalSection(This->pin.pCritSec); |
| |
| /* Flush outstanding samples */ |
| IAsyncReader_BeginFlush(This->pReader); |
| |
| for (;;) |
| { |
| DWORD_PTR dwUser; |
| |
| pSample = NULL; |
| IAsyncReader_WaitForNext(This->pReader, 0, &pSample, &dwUser); |
| |
| if (!pSample) |
| break; |
| |
| assert(!IMediaSample_GetActualDataLength(pSample)); |
| |
| IMediaSample_Release(pSample); |
| } |
| |
| IAsyncReader_EndFlush(This->pReader); |
| |
| LeaveCriticalSection(This->pin.pCritSec); |
| } |
| } |
| |
| static void PullPin_Thread_Process(PullPin *This) |
| { |
| HRESULT hr; |
| IMediaSample * pSample = NULL; |
| ALLOCATOR_PROPERTIES allocProps; |
| |
| hr = IMemAllocator_GetProperties(This->pAlloc, &allocProps); |
| |
| This->cbAlign = allocProps.cbAlign; |
| |
| if (This->rtCurrent < This->rtStart) |
| This->rtCurrent = MEDIATIME_FROM_BYTES(ALIGNDOWN(BYTES_FROM_MEDIATIME(This->rtStart), This->cbAlign)); |
| |
| TRACE("Start\n"); |
| |
| if (This->rtCurrent >= This->rtStop) |
| { |
| IPin_EndOfStream(&This->pin.IPin_iface); |
| return; |
| } |
| |
| /* There is no sample in our buffer */ |
| hr = This->fnCustomRequest(This->pUserData); |
| |
| if (FAILED(hr)) |
| ERR("Request error: %x\n", hr); |
| |
| EnterCriticalSection(This->pin.pCritSec); |
| SetEvent(This->hEventStateChanged); |
| LeaveCriticalSection(This->pin.pCritSec); |
| |
| if (SUCCEEDED(hr)) |
| do |
| { |
| DWORD_PTR dwUser; |
| |
| TRACE("Process sample\n"); |
| |
| pSample = NULL; |
| hr = IAsyncReader_WaitForNext(This->pReader, 10000, &pSample, &dwUser); |
| |
| /* Return an empty sample on error to the implementation in case it does custom parsing, so it knows it's gone */ |
| if (SUCCEEDED(hr)) |
| { |
| hr = This->fnSampleProc(This->pUserData, pSample, dwUser); |
| } |
| else |
| { |
| if (hr == VFW_E_TIMEOUT) |
| { |
| if (pSample != NULL) |
| WARN("Non-NULL sample returned with VFW_E_TIMEOUT.\n"); |
| hr = S_OK; |
| } |
| /* FIXME: Errors are not well handled yet! */ |
| else |
| ERR("Processing error: %x\n", hr); |
| } |
| |
| if (pSample) |
| { |
| IMediaSample_Release(pSample); |
| pSample = NULL; |
| } |
| } while (This->rtCurrent < This->rtStop && hr == S_OK && !This->stop_playback); |
| |
| /* |
| * Sample was rejected, and we are asked to terminate. When there is more than one buffer |
| * it is possible for a filter to have several queued samples, making it necessary to |
| * release all of these pending samples. |
| */ |
| if (This->stop_playback || FAILED(hr)) |
| { |
| DWORD_PTR dwUser; |
| |
| do |
| { |
| if (pSample) |
| IMediaSample_Release(pSample); |
| pSample = NULL; |
| IAsyncReader_WaitForNext(This->pReader, 0, &pSample, &dwUser); |
| } while(pSample); |
| } |
| |
| /* Can't reset state to Sleepy here because that might race, instead PauseProcessing will do that for us |
| * Flush remaining samples |
| */ |
| if (This->fnDone) |
| This->fnDone(This->pUserData); |
| |
| TRACE("End: %08x, %d\n", hr, This->stop_playback); |
| } |
| |
| static void PullPin_Thread_Pause(PullPin *This) |
| { |
| PullPin_Flush(This); |
| |
| EnterCriticalSection(This->pin.pCritSec); |
| This->state = Req_Sleepy; |
| SetEvent(This->hEventStateChanged); |
| LeaveCriticalSection(This->pin.pCritSec); |
| } |
| |
| static void PullPin_Thread_Stop(PullPin *This) |
| { |
| TRACE("(%p)->()\n", This); |
| |
| EnterCriticalSection(This->pin.pCritSec); |
| { |
| CloseHandle(This->hThread); |
| This->hThread = NULL; |
| SetEvent(This->hEventStateChanged); |
| } |
| LeaveCriticalSection(This->pin.pCritSec); |
| |
| IBaseFilter_Release(This->pin.pinInfo.pFilter); |
| |
| CoUninitialize(); |
| ExitThread(0); |
| } |
| |
| static DWORD WINAPI PullPin_Thread_Main(LPVOID pv) |
| { |
| PullPin *This = pv; |
| CoInitializeEx(NULL, COINIT_MULTITHREADED); |
| |
| PullPin_Flush(This); |
| |
| for (;;) |
| { |
| WaitForSingleObject(This->thread_sleepy, INFINITE); |
| |
| TRACE("State: %d\n", This->state); |
| |
| switch (This->state) |
| { |
| case Req_Die: PullPin_Thread_Stop(This); break; |
| case Req_Run: PullPin_Thread_Process(This); break; |
| case Req_Pause: PullPin_Thread_Pause(This); break; |
| case Req_Sleepy: ERR("Should not be signalled with SLEEPY!\n"); break; |
| default: ERR("Unknown state request: %d\n", This->state); break; |
| } |
| } |
| return 0; |
| } |
| |
| static HRESULT PullPin_InitProcessing(PullPin * This) |
| { |
| HRESULT hr = S_OK; |
| |
| TRACE("(%p)->()\n", This); |
| |
| /* if we are connected */ |
| if (This->pAlloc) |
| { |
| DWORD dwThreadId; |
| |
| WaitForSingleObject(This->hEventStateChanged, INFINITE); |
| EnterCriticalSection(This->pin.pCritSec); |
| |
| assert(!This->hThread); |
| assert(This->state == Req_Die); |
| assert(This->stop_playback); |
| assert(WaitForSingleObject(This->thread_sleepy, 0) == WAIT_TIMEOUT); |
| This->state = Req_Sleepy; |
| |
| /* AddRef the filter to make sure it and its pins will be around |
| * as long as the thread */ |
| IBaseFilter_AddRef(This->pin.pinInfo.pFilter); |
| |
| |
| This->hThread = CreateThread(NULL, 0, PullPin_Thread_Main, This, 0, &dwThreadId); |
| if (!This->hThread) |
| { |
| hr = HRESULT_FROM_WIN32(GetLastError()); |
| IBaseFilter_Release(This->pin.pinInfo.pFilter); |
| } |
| |
| if (SUCCEEDED(hr)) |
| { |
| SetEvent(This->hEventStateChanged); |
| /* If assert fails, that means a command was not processed before the thread previously terminated */ |
| } |
| LeaveCriticalSection(This->pin.pCritSec); |
| } |
| |
| TRACE(" -- %x\n", hr); |
| |
| return hr; |
| } |
| |
| HRESULT PullPin_StartProcessing(PullPin * This) |
| { |
| /* if we are connected */ |
| TRACE("(%p)->()\n", This); |
| if(This->pAlloc) |
| { |
| assert(This->hThread); |
| |
| PullPin_WaitForStateChange(This, INFINITE); |
| |
| assert(This->state == Req_Sleepy); |
| |
| /* Wake up! */ |
| assert(WaitForSingleObject(This->thread_sleepy, 0) == WAIT_TIMEOUT); |
| This->state = Req_Run; |
| This->stop_playback = FALSE; |
| ResetEvent(This->hEventStateChanged); |
| SetEvent(This->thread_sleepy); |
| } |
| |
| return S_OK; |
| } |
| |
| HRESULT PullPin_PauseProcessing(PullPin * This) |
| { |
| /* if we are connected */ |
| TRACE("(%p)->()\n", This); |
| if(This->pAlloc) |
| { |
| assert(This->hThread); |
| |
| PullPin_WaitForStateChange(This, INFINITE); |
| |
| EnterCriticalSection(This->pin.pCritSec); |
| |
| assert(!This->stop_playback); |
| assert(This->state == Req_Run|| This->state == Req_Sleepy); |
| |
| assert(WaitForSingleObject(This->thread_sleepy, 0) == WAIT_TIMEOUT); |
| |
| This->state = Req_Pause; |
| This->stop_playback = TRUE; |
| ResetEvent(This->hEventStateChanged); |
| SetEvent(This->thread_sleepy); |
| |
| /* Release any outstanding samples */ |
| if (This->pReader) |
| { |
| IMediaSample *pSample; |
| DWORD_PTR dwUser; |
| |
| do |
| { |
| pSample = NULL; |
| IAsyncReader_WaitForNext(This->pReader, 0, &pSample, &dwUser); |
| if (pSample) |
| IMediaSample_Release(pSample); |
| } while(pSample); |
| } |
| |
| LeaveCriticalSection(This->pin.pCritSec); |
| } |
| |
| return S_OK; |
| } |
| |
| static HRESULT PullPin_StopProcessing(PullPin * This) |
| { |
| TRACE("(%p)->()\n", This); |
| |
| /* if we are alive */ |
| assert(This->hThread); |
| |
| PullPin_WaitForStateChange(This, INFINITE); |
| |
| assert(This->state == Req_Pause || This->state == Req_Sleepy); |
| |
| This->stop_playback = TRUE; |
| This->state = Req_Die; |
| assert(WaitForSingleObject(This->thread_sleepy, 0) == WAIT_TIMEOUT); |
| ResetEvent(This->hEventStateChanged); |
| SetEvent(This->thread_sleepy); |
| return S_OK; |
| } |
| |
| HRESULT PullPin_WaitForStateChange(PullPin * This, DWORD dwMilliseconds) |
| { |
| if (WaitForSingleObject(This->hEventStateChanged, dwMilliseconds) == WAIT_TIMEOUT) |
| return S_FALSE; |
| return S_OK; |
| } |
| |
| HRESULT WINAPI PullPin_QueryAccept(IPin * iface, const AM_MEDIA_TYPE * pmt) |
| { |
| PullPin *This = impl_PullPin_from_IPin(iface); |
| |
| TRACE("(%p/%p)->(%p)\n", This, iface, pmt); |
| |
| return (This->fnQueryAccept(This->pUserData, pmt) == S_OK ? S_OK : S_FALSE); |
| } |
| |
| HRESULT WINAPI PullPin_EndOfStream(IPin * iface) |
| { |
| PullPin *This = impl_PullPin_from_IPin(iface); |
| HRESULT hr = S_FALSE; |
| |
| TRACE("(%p)->()\n", iface); |
| |
| EnterCriticalSection(This->pin.pCritSec); |
| hr = SendFurther( iface, deliver_endofstream, NULL, NULL ); |
| SetEvent(This->hEventStateChanged); |
| LeaveCriticalSection(This->pin.pCritSec); |
| |
| return hr; |
| } |
| |
| HRESULT WINAPI PullPin_BeginFlush(IPin * iface) |
| { |
| PullPin *This = impl_PullPin_from_IPin(iface); |
| TRACE("(%p)->()\n", This); |
| |
| EnterCriticalSection(This->pin.pCritSec); |
| { |
| SendFurther( iface, deliver_beginflush, NULL, NULL ); |
| } |
| LeaveCriticalSection(This->pin.pCritSec); |
| |
| EnterCriticalSection(&This->thread_lock); |
| { |
| if (This->pReader) |
| IAsyncReader_BeginFlush(This->pReader); |
| PullPin_WaitForStateChange(This, INFINITE); |
| |
| if (This->hThread && This->state == Req_Run) |
| { |
| PullPin_PauseProcessing(This); |
| PullPin_WaitForStateChange(This, INFINITE); |
| } |
| } |
| LeaveCriticalSection(&This->thread_lock); |
| |
| EnterCriticalSection(This->pin.pCritSec); |
| { |
| This->fnCleanProc(This->pUserData); |
| } |
| LeaveCriticalSection(This->pin.pCritSec); |
| |
| return S_OK; |
| } |
| |
| HRESULT WINAPI PullPin_EndFlush(IPin * iface) |
| { |
| PullPin *This = impl_PullPin_from_IPin(iface); |
| |
| TRACE("(%p)->()\n", iface); |
| |
| /* Send further first: Else a race condition might terminate processing early */ |
| EnterCriticalSection(This->pin.pCritSec); |
| SendFurther( iface, deliver_endflush, NULL, NULL ); |
| LeaveCriticalSection(This->pin.pCritSec); |
| |
| EnterCriticalSection(&This->thread_lock); |
| { |
| FILTER_STATE state; |
| |
| if (This->pReader) |
| IAsyncReader_EndFlush(This->pReader); |
| |
| IBaseFilter_GetState(This->pin.pinInfo.pFilter, INFINITE, &state); |
| |
| if (state != State_Stopped) |
| PullPin_StartProcessing(This); |
| |
| PullPin_WaitForStateChange(This, INFINITE); |
| } |
| LeaveCriticalSection(&This->thread_lock); |
| |
| return S_OK; |
| } |
| |
| HRESULT WINAPI PullPin_Disconnect(IPin *iface) |
| { |
| HRESULT hr; |
| PullPin *This = impl_PullPin_from_IPin(iface); |
| |
| TRACE("()\n"); |
| |
| EnterCriticalSection(This->pin.pCritSec); |
| { |
| if (FAILED(hr = IMemAllocator_Decommit(This->pAlloc))) |
| ERR("Allocator decommit failed with error %x. Possible memory leak\n", hr); |
| |
| if (This->pin.pConnectedTo) |
| { |
| IPin_Release(This->pin.pConnectedTo); |
| This->pin.pConnectedTo = NULL; |
| PullPin_StopProcessing(This); |
| |
| FreeMediaType(&This->pin.mtCurrent); |
| ZeroMemory(&This->pin.mtCurrent, sizeof(This->pin.mtCurrent)); |
| hr = S_OK; |
| } |
| else |
| hr = S_FALSE; |
| } |
| LeaveCriticalSection(This->pin.pCritSec); |
| |
| return hr; |
| } |
| |
| HRESULT WINAPI PullPin_NewSegment(IPin * iface, REFERENCE_TIME tStart, REFERENCE_TIME tStop, double dRate) |
| { |
| newsegmentargs args; |
| FIXME("(%p)->(%s, %s, %g) stub\n", iface, wine_dbgstr_longlong(tStart), wine_dbgstr_longlong(tStop), dRate); |
| |
| args.tStart = tStart; |
| args.tStop = tStop; |
| args.rate = dRate; |
| |
| return SendFurther( iface, deliver_newsegment, &args, NULL ); |
| } |