strmbase: Implement BaseFilter in strmbase.
[wine/wine-gecko.git] / dlls / strmbase / enumpins.c
blob53a1e00095093ce46bf34841b38be562dc124d70
1 /*
2 * Implementation of IEnumPins Interface
4 * Copyright 2003 Robert Shearman
5 * Copyright 2010 Aric Stewart, CodeWeavers
7 * This library is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Lesser General Public
9 * License as published by the Free Software Foundation; either
10 * version 2.1 of the License, or (at your option) any later version.
12 * This library is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Lesser General Public License for more details.
17 * You should have received a copy of the GNU Lesser General Public
18 * License along with this library; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
22 #define COBJMACROS
24 #include "dshow.h"
25 #include "wine/strmbase.h"
26 #include "wine/debug.h"
28 WINE_DEFAULT_DEBUG_CHANNEL(strmbase);
30 typedef struct IEnumPinsImpl
32 const IEnumPinsVtbl * lpVtbl;
33 LONG refCount;
34 ULONG uIndex;
35 IBaseFilter *base;
36 BaseFilter_GetPin receive_pin;
37 BaseFilter_GetPinCount receive_pincount;
38 BaseFilter_GetPinVersion receive_version;
39 DWORD Version;
40 } IEnumPinsImpl;
42 static const struct IEnumPinsVtbl IEnumPinsImpl_Vtbl;
44 HRESULT WINAPI EnumPins_Construct(IBaseFilter *base, BaseFilter_GetPin receive_pin, BaseFilter_GetPinCount receive_pincount, BaseFilter_GetPinVersion receive_version, IEnumPins ** ppEnum)
46 IEnumPinsImpl * pEnumPins;
48 if (!ppEnum)
49 return E_POINTER;
51 pEnumPins = CoTaskMemAlloc(sizeof(IEnumPinsImpl));
52 if (!pEnumPins)
54 *ppEnum = NULL;
55 return E_OUTOFMEMORY;
57 pEnumPins->lpVtbl = &IEnumPinsImpl_Vtbl;
58 pEnumPins->refCount = 1;
59 pEnumPins->uIndex = 0;
60 pEnumPins->receive_pin = receive_pin;
61 pEnumPins->receive_pincount = receive_pincount;
62 pEnumPins->receive_version = receive_version;
63 pEnumPins->base = base;
64 IBaseFilter_AddRef(base);
65 *ppEnum = (IEnumPins *)(&pEnumPins->lpVtbl);
66 pEnumPins->Version = receive_version(base);
68 TRACE("Created new enumerator (%p)\n", *ppEnum);
69 return S_OK;
72 static HRESULT WINAPI IEnumPinsImpl_QueryInterface(IEnumPins * iface, REFIID riid, LPVOID * ppv)
74 TRACE("(%s, %p)\n", debugstr_guid(riid), ppv);
76 *ppv = NULL;
78 if (IsEqualIID(riid, &IID_IUnknown))
79 *ppv = iface;
80 else if (IsEqualIID(riid, &IID_IEnumPins))
81 *ppv = iface;
83 if (*ppv)
85 IUnknown_AddRef((IUnknown *)(*ppv));
86 return S_OK;
89 FIXME("No interface for %s!\n", debugstr_guid(riid));
91 return E_NOINTERFACE;
94 static ULONG WINAPI IEnumPinsImpl_AddRef(IEnumPins * iface)
96 IEnumPinsImpl *This = (IEnumPinsImpl *)iface;
97 ULONG refCount = InterlockedIncrement(&This->refCount);
99 TRACE("(%p)->() AddRef from %d\n", This, refCount - 1);
101 return refCount;
104 static ULONG WINAPI IEnumPinsImpl_Release(IEnumPins * iface)
106 IEnumPinsImpl *This = (IEnumPinsImpl *)iface;
107 ULONG refCount = InterlockedDecrement(&This->refCount);
109 TRACE("(%p)->() Release from %d\n", This, refCount + 1);
111 if (!refCount)
113 IBaseFilter_Release(This->base);
114 CoTaskMemFree(This);
115 return 0;
117 else
118 return refCount;
121 static HRESULT WINAPI IEnumPinsImpl_Next(IEnumPins * iface, ULONG cPins, IPin ** ppPins, ULONG * pcFetched)
123 IEnumPinsImpl *This = (IEnumPinsImpl *)iface;
124 HRESULT hr = S_OK;
125 ULONG i = 0;
127 TRACE("(%u, %p, %p)\n", cPins, ppPins, pcFetched);
129 if (!ppPins)
130 return E_POINTER;
132 if (cPins > 1 && !pcFetched)
133 return E_INVALIDARG;
135 if (pcFetched)
136 *pcFetched = 0;
138 if (This->Version != This->receive_version(This->base))
139 return VFW_E_ENUM_OUT_OF_SYNC;
141 while (i < cPins && hr == S_OK)
143 IPin *pin;
144 pin = This->receive_pin(This->base, This->uIndex + i);
146 if (!pin)
147 break;
148 else
149 ppPins[i] = pin;
150 ++i;
153 if (pcFetched)
154 *pcFetched = i;
155 This->uIndex += i;
157 if (i < cPins)
158 return S_FALSE;
159 return S_OK;
162 static HRESULT WINAPI IEnumPinsImpl_Skip(IEnumPins * iface, ULONG cPins)
164 IEnumPinsImpl *This = (IEnumPinsImpl *)iface;
166 TRACE("(%u)\n", cPins);
168 if (This->Version != This->receive_version(This->base))
169 return VFW_E_ENUM_OUT_OF_SYNC;
171 if (This->receive_pincount(This->base) >= This->uIndex + cPins)
172 return S_FALSE;
174 This->uIndex += cPins;
175 return S_OK;
178 static HRESULT WINAPI IEnumPinsImpl_Reset(IEnumPins * iface)
180 IEnumPinsImpl *This = (IEnumPinsImpl *)iface;
182 TRACE("IEnumPinsImpl::Reset()\n");
183 This->Version = This->receive_version(This->base);
185 This->uIndex = 0;
186 return S_OK;
189 static HRESULT WINAPI IEnumPinsImpl_Clone(IEnumPins * iface, IEnumPins ** ppEnum)
191 HRESULT hr;
192 IEnumPinsImpl *This = (IEnumPinsImpl *)iface;
194 TRACE("(%p)\n", ppEnum);
196 hr = EnumPins_Construct(This->base, This->receive_pin, This->receive_pincount, This->receive_version, ppEnum);
197 if (FAILED(hr))
198 return hr;
199 return IEnumPins_Skip(*ppEnum, This->uIndex);
202 static const IEnumPinsVtbl IEnumPinsImpl_Vtbl =
204 IEnumPinsImpl_QueryInterface,
205 IEnumPinsImpl_AddRef,
206 IEnumPinsImpl_Release,
207 IEnumPinsImpl_Next,
208 IEnumPinsImpl_Skip,
209 IEnumPinsImpl_Reset,
210 IEnumPinsImpl_Clone