amstream: Implement IAMMultiMediaStreamImpl_Seek.
[wine/multimedia.git] / dlls / amstream / amstream.c
blob1366cdba7d29d710519360652bca80f89f26338d
1 /*
2 * Implementation of IAMMultiMediaStream Interface
4 * Copyright 2004, 2012 Christian Costa
5 * Copyright 2006 Ivan Leo Puoti
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 #include "wine/debug.h"
24 #define COBJMACROS
26 #include "winbase.h"
27 #include "wingdi.h"
29 #include "amstream_private.h"
30 #include "amstream.h"
32 WINE_DEFAULT_DEBUG_CHANNEL(amstream);
34 typedef struct {
35 IAMMultiMediaStream IAMMultiMediaStream_iface;
36 LONG ref;
37 IGraphBuilder* pFilterGraph;
38 IMediaSeeking* media_seeking;
39 IMediaControl* media_control;
40 IBaseFilter* media_stream_filter;
41 IPin* ipin;
42 ULONG nbStreams;
43 IMediaStream** pStreams;
44 STREAM_TYPE StreamType;
45 OAEVENT event;
46 } IAMMultiMediaStreamImpl;
48 static inline IAMMultiMediaStreamImpl *impl_from_IAMMultiMediaStream(IAMMultiMediaStream *iface)
50 return CONTAINING_RECORD(iface, IAMMultiMediaStreamImpl, IAMMultiMediaStream_iface);
53 static const struct IAMMultiMediaStreamVtbl AM_Vtbl;
55 HRESULT AM_create(IUnknown *pUnkOuter, LPVOID *ppObj)
57 IAMMultiMediaStreamImpl* object;
59 TRACE("(%p,%p)\n", pUnkOuter, ppObj);
61 if( pUnkOuter )
62 return CLASS_E_NOAGGREGATION;
64 object = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IAMMultiMediaStreamImpl));
65 if (!object)
67 ERR("Out of memory\n");
68 return E_OUTOFMEMORY;
71 object->IAMMultiMediaStream_iface.lpVtbl = &AM_Vtbl;
72 object->ref = 1;
74 *ppObj = object;
76 return S_OK;
79 /*** IUnknown methods ***/
80 static HRESULT WINAPI IAMMultiMediaStreamImpl_QueryInterface(IAMMultiMediaStream* iface, REFIID riid, void** ppvObject)
82 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
84 TRACE("(%p/%p)->(%s,%p)\n", iface, This, debugstr_guid(riid), ppvObject);
86 if (IsEqualGUID(riid, &IID_IUnknown) ||
87 IsEqualGUID(riid, &IID_IMultiMediaStream) ||
88 IsEqualGUID(riid, &IID_IAMMultiMediaStream))
90 IUnknown_AddRef(iface);
91 *ppvObject = This;
92 return S_OK;
95 ERR("(%p)->(%s,%p),not found\n",This,debugstr_guid(riid),ppvObject);
97 return E_NOINTERFACE;
100 static ULONG WINAPI IAMMultiMediaStreamImpl_AddRef(IAMMultiMediaStream* iface)
102 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
104 TRACE("(%p/%p)\n", iface, This);
106 return InterlockedIncrement(&This->ref);
109 static ULONG WINAPI IAMMultiMediaStreamImpl_Release(IAMMultiMediaStream* iface)
111 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
112 ULONG ref = InterlockedDecrement(&This->ref);
113 ULONG i;
115 TRACE("(%p/%p)\n", iface, This);
117 if (!ref)
119 for(i = 0; i < This->nbStreams; i++)
120 IMediaStream_Release(This->pStreams[i]);
121 if (This->ipin)
122 IPin_Release(This->ipin);
123 if (This->media_stream_filter)
124 IBaseFilter_Release(This->media_stream_filter);
125 if (This->media_seeking)
126 IMediaSeeking_Release(This->media_seeking);
127 if (This->media_control)
128 IMediaControl_Release(This->media_control);
129 if (This->pFilterGraph)
130 IGraphBuilder_Release(This->pFilterGraph);
131 HeapFree(GetProcessHeap(), 0, This);
134 return ref;
137 /*** IMultiMediaStream methods ***/
138 static HRESULT WINAPI IAMMultiMediaStreamImpl_GetInformation(IAMMultiMediaStream* iface, DWORD* pdwFlags, STREAM_TYPE* pStreamType)
140 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
142 FIXME("(%p/%p)->(%p,%p) stub!\n", This, iface, pdwFlags, pStreamType);
144 return E_NOTIMPL;
147 static HRESULT WINAPI IAMMultiMediaStreamImpl_GetMediaStream(IAMMultiMediaStream* iface, REFMSPID idPurpose, IMediaStream** ppMediaStream)
149 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
150 MSPID PurposeId;
151 unsigned int i;
153 TRACE("(%p/%p)->(%s,%p)\n", This, iface, debugstr_guid(idPurpose), ppMediaStream);
155 for (i = 0; i < This->nbStreams; i++)
157 IMediaStream_GetInformation(This->pStreams[i], &PurposeId, NULL);
158 if (IsEqualIID(&PurposeId, idPurpose))
160 *ppMediaStream = This->pStreams[i];
161 IMediaStream_AddRef(*ppMediaStream);
162 return S_OK;
166 return MS_E_NOSTREAM;
169 static HRESULT WINAPI IAMMultiMediaStreamImpl_EnumMediaStreams(IAMMultiMediaStream* iface, LONG Index, IMediaStream** ppMediaStream)
171 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
173 FIXME("(%p/%p)->(%d,%p) stub!\n", This, iface, Index, ppMediaStream);
175 return E_NOTIMPL;
178 static HRESULT WINAPI IAMMultiMediaStreamImpl_GetState(IAMMultiMediaStream* iface, STREAM_STATE* pCurrentState)
180 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
182 FIXME("(%p/%p)->(%p) stub!\n", This, iface, pCurrentState);
184 return E_NOTIMPL;
187 static HRESULT WINAPI IAMMultiMediaStreamImpl_SetState(IAMMultiMediaStream* iface, STREAM_STATE NewState)
189 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
191 FIXME("(%p/%p)->() stub!\n", This, iface);
193 return E_NOTIMPL;
196 static HRESULT WINAPI IAMMultiMediaStreamImpl_GetTime(IAMMultiMediaStream* iface, STREAM_TIME* pCurrentTime)
198 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
200 FIXME("(%p/%p)->(%p) stub!\n", This, iface, pCurrentTime);
202 return E_NOTIMPL;
205 static HRESULT WINAPI IAMMultiMediaStreamImpl_GetDuration(IAMMultiMediaStream* iface, STREAM_TIME* pDuration)
207 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
209 FIXME("(%p/%p)->(%p) stub!\n", This, iface, pDuration);
211 return E_NOTIMPL;
214 static HRESULT WINAPI IAMMultiMediaStreamImpl_Seek(IAMMultiMediaStream* iface, STREAM_TIME seek_time)
216 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
218 TRACE("(%p/%p)->(%s)\n", This, iface, wine_dbgstr_longlong(seek_time));
220 return IMediaSeeking_SetPositions(This->media_seeking, &seek_time, AM_SEEKING_AbsolutePositioning, NULL, AM_SEEKING_NoPositioning);
223 static HRESULT WINAPI IAMMultiMediaStreamImpl_GetEndOfStream(IAMMultiMediaStream* iface, HANDLE* phEOS)
225 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
227 FIXME("(%p/%p)->(%p) stub!\n", This, iface, phEOS);
229 return E_NOTIMPL;
232 /*** IAMMultiMediaStream methods ***/
233 static HRESULT WINAPI IAMMultiMediaStreamImpl_Initialize(IAMMultiMediaStream* iface, STREAM_TYPE StreamType, DWORD dwFlags, IGraphBuilder* pFilterGraph)
235 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
236 HRESULT hr = S_OK;
237 const WCHAR filternameW[] = {'M','e','d','i','a','S','t','r','e','a','m','F','i','l','t','e','r',0};
239 TRACE("(%p/%p)->(%x,%x,%p)\n", This, iface, (DWORD)StreamType, dwFlags, pFilterGraph);
241 if (pFilterGraph)
243 This->pFilterGraph = pFilterGraph;
244 IGraphBuilder_AddRef(This->pFilterGraph);
246 else
248 hr = CoCreateInstance(&CLSID_FilterGraph, NULL, CLSCTX_INPROC_SERVER, &IID_IGraphBuilder, (LPVOID*)&This->pFilterGraph);
251 if (SUCCEEDED(hr))
253 This->StreamType = StreamType;
254 hr = IGraphBuilder_QueryInterface(This->pFilterGraph, &IID_IMediaSeeking, (void**)&This->media_seeking);
255 if (SUCCEEDED(hr))
256 IGraphBuilder_QueryInterface(This->pFilterGraph, &IID_IMediaControl, (void**)&This->media_control);
257 if (SUCCEEDED(hr))
258 hr = CoCreateInstance(&CLSID_MediaStreamFilter, NULL, CLSCTX_INPROC_SERVER, &IID_IBaseFilter, (LPVOID*)&This->media_stream_filter);
259 if (SUCCEEDED(hr))
260 IGraphBuilder_AddFilter(This->pFilterGraph, This->media_stream_filter, filternameW);
261 if (SUCCEEDED(hr))
263 IMediaEventEx* media_event = NULL;
264 hr = IGraphBuilder_QueryInterface(This->pFilterGraph, &IID_IMediaEventEx, (void**)&media_event);
265 if (SUCCEEDED(hr))
266 hr = IMediaEventEx_GetEventHandle(media_event, &This->event);
267 if (SUCCEEDED(hr))
268 hr = IMediaEventEx_SetNotifyFlags(media_event, AM_MEDIAEVENT_NONOTIFY);
269 if (media_event)
270 IMediaEventEx_Release(media_event);
274 if (FAILED(hr))
276 if (This->media_stream_filter)
277 IBaseFilter_Release(This->media_stream_filter);
278 This->media_stream_filter = NULL;
279 if (This->media_seeking)
280 IMediaSeeking_Release(This->media_seeking);
281 This->media_seeking = NULL;
282 if (This->media_control)
283 IMediaControl_Release(This->media_control);
284 This->media_control = NULL;
285 if (This->pFilterGraph)
286 IGraphBuilder_Release(This->pFilterGraph);
287 This->pFilterGraph = NULL;
290 return hr;
293 static HRESULT WINAPI IAMMultiMediaStreamImpl_GetFilterGraph(IAMMultiMediaStream* iface, IGraphBuilder** ppGraphBuilder)
295 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
297 TRACE("(%p/%p)->(%p)\n", This, iface, ppGraphBuilder);
299 if (!ppGraphBuilder)
300 return E_POINTER;
302 if (This->pFilterGraph)
303 return IFilterGraph_QueryInterface(This->pFilterGraph, &IID_IGraphBuilder, (void**)ppGraphBuilder);
304 else
305 *ppGraphBuilder = NULL;
307 return S_OK;
310 static HRESULT WINAPI IAMMultiMediaStreamImpl_GetFilter(IAMMultiMediaStream* iface, IMediaStreamFilter** ppFilter)
312 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
313 HRESULT hr = S_OK;
315 TRACE("(%p/%p)->(%p)\n", This, iface, ppFilter);
317 if (!ppFilter)
318 return E_POINTER;
320 *ppFilter = NULL;
322 if (This->media_stream_filter)
323 hr = IBaseFilter_QueryInterface(This->media_stream_filter, &IID_IMediaStreamFilter, (LPVOID*)ppFilter);
325 return hr;
328 static HRESULT WINAPI IAMMultiMediaStreamImpl_AddMediaStream(IAMMultiMediaStream* iface, IUnknown* pStreamObject, const MSPID* PurposeId,
329 DWORD dwFlags, IMediaStream** ppNewStream)
331 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
332 HRESULT hr;
333 IMediaStream* pStream;
334 IMediaStream** pNewStreams;
336 TRACE("(%p/%p)->(%p,%s,%x,%p)\n", This, iface, pStreamObject, debugstr_guid(PurposeId), dwFlags, ppNewStream);
338 if (!IsEqualGUID(PurposeId, &MSPID_PrimaryVideo) && !IsEqualGUID(PurposeId, &MSPID_PrimaryAudio))
339 return MS_E_PURPOSEID;
341 if (dwFlags & AMMSF_ADDDEFAULTRENDERER)
343 if (IsEqualGUID(PurposeId, &MSPID_PrimaryVideo))
345 /* Default renderer not supported by video stream */
346 return MS_E_PURPOSEID;
348 else
350 IBaseFilter* dsoundrender_filter;
352 /* Create the default renderer for audio */
353 hr = CoCreateInstance(&CLSID_DSoundRender, NULL, CLSCTX_INPROC_SERVER, &IID_IBaseFilter, (LPVOID*)&dsoundrender_filter);
354 if (SUCCEEDED(hr))
356 hr = IGraphBuilder_AddFilter(This->pFilterGraph, dsoundrender_filter, NULL);
357 IBaseFilter_Release(dsoundrender_filter);
360 /* No media stream created when the default renderer is used */
361 return hr;
365 hr = mediastream_create((IMultiMediaStream*)iface, PurposeId, This->StreamType, &pStream);
366 if (SUCCEEDED(hr))
368 pNewStreams = CoTaskMemRealloc(This->pStreams, (This->nbStreams+1) * sizeof(IMediaStream*));
369 if (!pNewStreams)
371 IMediaStream_Release(pStream);
372 return E_OUTOFMEMORY;
374 This->pStreams = pNewStreams;
375 This->pStreams[This->nbStreams] = pStream;
376 This->nbStreams++;
378 if (ppNewStream)
379 *ppNewStream = pStream;
382 if (SUCCEEDED(hr))
384 /* Add stream to the media stream filter */
385 IMediaStreamFilter_AddMediaStream((IMediaStreamFilter*)This->media_stream_filter, (IAMMediaStream*)pStream);
388 return hr;
391 static HRESULT WINAPI IAMMultiMediaStreamImpl_OpenFile(IAMMultiMediaStream* iface, LPCWSTR pszFileName, DWORD dwFlags)
393 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
394 HRESULT ret = S_OK;
395 IBaseFilter *BaseFilter = NULL;
396 IEnumPins *EnumPins = NULL;
397 IPin *ipin;
398 PIN_DIRECTION pin_direction;
399 const WCHAR sourceW[] = {'S','o','u','r','c','e',0};
401 TRACE("(%p/%p)->(%s,%x)\n", This, iface, debugstr_w(pszFileName), dwFlags);
403 /* If Initialize was not called before, we do it here */
404 if (!This->pFilterGraph)
405 ret = IAMMultiMediaStream_Initialize(iface, STREAMTYPE_READ, 0, NULL);
407 if (SUCCEEDED(ret))
408 ret = IGraphBuilder_AddSourceFilter(This->pFilterGraph, pszFileName, sourceW, &BaseFilter);
410 if (SUCCEEDED(ret))
411 ret = IBaseFilter_EnumPins(BaseFilter, &EnumPins);
413 if (SUCCEEDED(ret))
414 ret = IEnumPins_Next(EnumPins, 1, &ipin, NULL);
416 if (SUCCEEDED(ret))
418 ret = IPin_QueryDirection(ipin, &pin_direction);
419 if (ret == S_OK && pin_direction == PINDIR_OUTPUT)
420 This->ipin = ipin;
423 if (EnumPins)
424 IEnumPins_Release(EnumPins);
425 if (BaseFilter)
426 IBaseFilter_Release(BaseFilter);
427 return ret;
430 static HRESULT WINAPI IAMMultiMediaStreamImpl_OpenMoniker(IAMMultiMediaStream* iface, IBindCtx* pCtx, IMoniker* pMoniker, DWORD dwFlags)
432 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
434 FIXME("(%p/%p)->(%p,%p,%x) stub!\n", This, iface, pCtx, pMoniker, dwFlags);
436 return E_NOTIMPL;
439 static HRESULT WINAPI IAMMultiMediaStreamImpl_Render(IAMMultiMediaStream* iface, DWORD dwFlags)
441 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
443 FIXME("(%p/%p)->(%x) partial stub!\n", This, iface, dwFlags);
445 if(dwFlags != AMMSF_NOCLOCK)
446 return E_INVALIDARG;
448 return IGraphBuilder_Render(This->pFilterGraph, This->ipin);
451 static const IAMMultiMediaStreamVtbl AM_Vtbl =
453 IAMMultiMediaStreamImpl_QueryInterface,
454 IAMMultiMediaStreamImpl_AddRef,
455 IAMMultiMediaStreamImpl_Release,
456 IAMMultiMediaStreamImpl_GetInformation,
457 IAMMultiMediaStreamImpl_GetMediaStream,
458 IAMMultiMediaStreamImpl_EnumMediaStreams,
459 IAMMultiMediaStreamImpl_GetState,
460 IAMMultiMediaStreamImpl_SetState,
461 IAMMultiMediaStreamImpl_GetTime,
462 IAMMultiMediaStreamImpl_GetDuration,
463 IAMMultiMediaStreamImpl_Seek,
464 IAMMultiMediaStreamImpl_GetEndOfStream,
465 IAMMultiMediaStreamImpl_Initialize,
466 IAMMultiMediaStreamImpl_GetFilterGraph,
467 IAMMultiMediaStreamImpl_GetFilter,
468 IAMMultiMediaStreamImpl_AddMediaStream,
469 IAMMultiMediaStreamImpl_OpenFile,
470 IAMMultiMediaStreamImpl_OpenMoniker,
471 IAMMultiMediaStreamImpl_Render