|  | /* | 
|  | * Implementation of IEnumPins 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 "wine/debug.h" | 
|  |  | 
|  | WINE_DEFAULT_DEBUG_CHANNEL(quartz); | 
|  |  | 
|  | typedef struct IEnumPinsImpl | 
|  | { | 
|  | const IEnumPinsVtbl * lpVtbl; | 
|  | LONG refCount; | 
|  | ULONG uIndex; | 
|  | IBaseFilter *base; | 
|  | FNOBTAINPIN receive_pin; | 
|  | DWORD synctime; | 
|  | } IEnumPinsImpl; | 
|  |  | 
|  | static const struct IEnumPinsVtbl IEnumPinsImpl_Vtbl; | 
|  |  | 
|  | HRESULT IEnumPinsImpl_Construct(IEnumPins ** ppEnum, FNOBTAINPIN receive_pin, IBaseFilter *base) | 
|  | { | 
|  | IEnumPinsImpl * pEnumPins; | 
|  |  | 
|  | if (!ppEnum) | 
|  | return E_POINTER; | 
|  |  | 
|  | pEnumPins = CoTaskMemAlloc(sizeof(IEnumPinsImpl)); | 
|  | if (!pEnumPins) | 
|  | { | 
|  | *ppEnum = NULL; | 
|  | return E_OUTOFMEMORY; | 
|  | } | 
|  | pEnumPins->lpVtbl = &IEnumPinsImpl_Vtbl; | 
|  | pEnumPins->refCount = 1; | 
|  | pEnumPins->uIndex = 0; | 
|  | pEnumPins->receive_pin = receive_pin; | 
|  | pEnumPins->base = base; | 
|  | IBaseFilter_AddRef(base); | 
|  | *ppEnum = (IEnumPins *)(&pEnumPins->lpVtbl); | 
|  |  | 
|  | receive_pin(base, ~0, NULL, &pEnumPins->synctime); | 
|  |  | 
|  | TRACE("Created new enumerator (%p)\n", *ppEnum); | 
|  | return S_OK; | 
|  | } | 
|  |  | 
|  | static HRESULT WINAPI IEnumPinsImpl_QueryInterface(IEnumPins * iface, REFIID riid, LPVOID * ppv) | 
|  | { | 
|  | TRACE("(%s, %p)\n", qzdebugstr_guid(riid), ppv); | 
|  |  | 
|  | *ppv = NULL; | 
|  |  | 
|  | if (IsEqualIID(riid, &IID_IUnknown)) | 
|  | *ppv = iface; | 
|  | else if (IsEqualIID(riid, &IID_IEnumPins)) | 
|  | *ppv = iface; | 
|  |  | 
|  | if (*ppv) | 
|  | { | 
|  | IUnknown_AddRef((IUnknown *)(*ppv)); | 
|  | return S_OK; | 
|  | } | 
|  |  | 
|  | FIXME("No interface for %s!\n", qzdebugstr_guid(riid)); | 
|  |  | 
|  | return E_NOINTERFACE; | 
|  | } | 
|  |  | 
|  | static ULONG WINAPI IEnumPinsImpl_AddRef(IEnumPins * iface) | 
|  | { | 
|  | IEnumPinsImpl *This = (IEnumPinsImpl *)iface; | 
|  | ULONG refCount = InterlockedIncrement(&This->refCount); | 
|  |  | 
|  | TRACE("(%p)->() AddRef from %d\n", This, refCount - 1); | 
|  |  | 
|  | return refCount; | 
|  | } | 
|  |  | 
|  | static ULONG WINAPI IEnumPinsImpl_Release(IEnumPins * iface) | 
|  | { | 
|  | IEnumPinsImpl *This = (IEnumPinsImpl *)iface; | 
|  | ULONG refCount = InterlockedDecrement(&This->refCount); | 
|  |  | 
|  | TRACE("(%p)->() Release from %d\n", This, refCount + 1); | 
|  |  | 
|  | if (!refCount) | 
|  | { | 
|  | IBaseFilter_Release(This->base); | 
|  | CoTaskMemFree(This); | 
|  | return 0; | 
|  | } | 
|  | else | 
|  | return refCount; | 
|  | } | 
|  |  | 
|  | static HRESULT WINAPI IEnumPinsImpl_Next(IEnumPins * iface, ULONG cPins, IPin ** ppPins, ULONG * pcFetched) | 
|  | { | 
|  | IEnumPinsImpl *This = (IEnumPinsImpl *)iface; | 
|  | DWORD synctime = This->synctime; | 
|  | HRESULT hr = S_OK; | 
|  | ULONG i = 0; | 
|  |  | 
|  | TRACE("(%u, %p, %p)\n", cPins, ppPins, pcFetched); | 
|  |  | 
|  | if (!ppPins) | 
|  | return E_POINTER; | 
|  |  | 
|  | if (cPins > 1 && !pcFetched) | 
|  | return E_INVALIDARG; | 
|  |  | 
|  | if (pcFetched) | 
|  | *pcFetched = 0; | 
|  |  | 
|  | while (i < cPins && hr == S_OK) | 
|  | { | 
|  | hr = This->receive_pin(This->base, This->uIndex + i, &ppPins[i], &synctime); | 
|  |  | 
|  | if (hr == S_OK) | 
|  | ++i; | 
|  |  | 
|  | if (synctime != This->synctime) | 
|  | break; | 
|  | } | 
|  |  | 
|  | if (!i && synctime != This->synctime) | 
|  | return VFW_E_ENUM_OUT_OF_SYNC; | 
|  |  | 
|  | if (pcFetched) | 
|  | *pcFetched = i; | 
|  | This->uIndex += i; | 
|  |  | 
|  | if (i < cPins) | 
|  | return S_FALSE; | 
|  | return S_OK; | 
|  | } | 
|  |  | 
|  | static HRESULT WINAPI IEnumPinsImpl_Skip(IEnumPins * iface, ULONG cPins) | 
|  | { | 
|  | IEnumPinsImpl *This = (IEnumPinsImpl *)iface; | 
|  | DWORD synctime = This->synctime; | 
|  | HRESULT hr; | 
|  | IPin *pin = NULL; | 
|  |  | 
|  | TRACE("(%u)\n", cPins); | 
|  |  | 
|  | hr = This->receive_pin(This->base, This->uIndex + cPins, &pin, &synctime); | 
|  | if (pin) | 
|  | IPin_Release(pin); | 
|  |  | 
|  | if (synctime != This->synctime) | 
|  | return VFW_E_ENUM_OUT_OF_SYNC; | 
|  |  | 
|  | if (hr == S_OK) | 
|  | This->uIndex += cPins; | 
|  |  | 
|  | return hr; | 
|  | } | 
|  |  | 
|  | static HRESULT WINAPI IEnumPinsImpl_Reset(IEnumPins * iface) | 
|  | { | 
|  | IEnumPinsImpl *This = (IEnumPinsImpl *)iface; | 
|  |  | 
|  | TRACE("IEnumPinsImpl::Reset()\n"); | 
|  | This->receive_pin(This->base, ~0, NULL, &This->synctime); | 
|  |  | 
|  | This->uIndex = 0; | 
|  | return S_OK; | 
|  | } | 
|  |  | 
|  | static HRESULT WINAPI IEnumPinsImpl_Clone(IEnumPins * iface, IEnumPins ** ppEnum) | 
|  | { | 
|  | HRESULT hr; | 
|  | IEnumPinsImpl *This = (IEnumPinsImpl *)iface; | 
|  |  | 
|  | TRACE("(%p)\n", ppEnum); | 
|  |  | 
|  | hr = IEnumPinsImpl_Construct(ppEnum, This->receive_pin, This->base); | 
|  | if (FAILED(hr)) | 
|  | return hr; | 
|  | return IEnumPins_Skip(*ppEnum, This->uIndex); | 
|  | } | 
|  |  | 
|  | static const IEnumPinsVtbl IEnumPinsImpl_Vtbl = | 
|  | { | 
|  | IEnumPinsImpl_QueryInterface, | 
|  | IEnumPinsImpl_AddRef, | 
|  | IEnumPinsImpl_Release, | 
|  | IEnumPinsImpl_Next, | 
|  | IEnumPinsImpl_Skip, | 
|  | IEnumPinsImpl_Reset, | 
|  | IEnumPinsImpl_Clone | 
|  | }; |