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"
29 #include "amstream_private.h"
32 WINE_DEFAULT_DEBUG_CHANNEL(amstream);
35 IAMMultiMediaStream IAMMultiMediaStream_iface;
37 IGraphBuilder* pFilterGraph;
38 IMediaSeeking* media_seeking;
39 IMediaControl* media_control;
40 IBaseFilter* media_stream_filter;
43 IMediaStream** pStreams;
44 STREAM_TYPE StreamType;
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);
62 return CLASS_E_NOAGGREGATION;
64 object = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IAMMultiMediaStreamImpl));
68 object->IAMMultiMediaStream_iface.lpVtbl = &AM_Vtbl;
76 /*** IUnknown methods ***/
77 static HRESULT WINAPI IAMMultiMediaStreamImpl_QueryInterface(IAMMultiMediaStream* iface, REFIID riid, void** ppvObject)
79 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
81 TRACE("(%p/%p)->(%s,%p)\n", iface, This, debugstr_guid(riid), ppvObject);
83 if (IsEqualGUID(riid, &IID_IUnknown) ||
84 IsEqualGUID(riid, &IID_IMultiMediaStream) ||
85 IsEqualGUID(riid, &IID_IAMMultiMediaStream))
87 IAMMultiMediaStream_AddRef(iface);
92 ERR("(%p)->(%s,%p),not found\n",This,debugstr_guid(riid),ppvObject);
97 static ULONG WINAPI IAMMultiMediaStreamImpl_AddRef(IAMMultiMediaStream* iface)
99 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
101 TRACE("(%p/%p)\n", iface, This);
103 return InterlockedIncrement(&This->ref);
106 static ULONG WINAPI IAMMultiMediaStreamImpl_Release(IAMMultiMediaStream* iface)
108 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
109 ULONG ref = InterlockedDecrement(&This->ref);
112 TRACE("(%p/%p)\n", iface, This);
116 for(i = 0; i < This->nbStreams; i++)
117 IMediaStream_Release(This->pStreams[i]);
119 IPin_Release(This->ipin);
120 if (This->media_stream_filter)
121 IBaseFilter_Release(This->media_stream_filter);
122 if (This->media_seeking)
123 IMediaSeeking_Release(This->media_seeking);
124 if (This->media_control)
125 IMediaControl_Release(This->media_control);
126 if (This->pFilterGraph)
127 IGraphBuilder_Release(This->pFilterGraph);
128 HeapFree(GetProcessHeap(), 0, This);
134 /*** IMultiMediaStream methods ***/
135 static HRESULT WINAPI IAMMultiMediaStreamImpl_GetInformation(IAMMultiMediaStream* iface, DWORD* pdwFlags, STREAM_TYPE* pStreamType)
137 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
139 FIXME("(%p/%p)->(%p,%p) stub!\n", This, iface, pdwFlags, pStreamType);
144 static HRESULT WINAPI IAMMultiMediaStreamImpl_GetMediaStream(IAMMultiMediaStream* iface, REFMSPID idPurpose, IMediaStream** ppMediaStream)
146 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
150 TRACE("(%p/%p)->(%s,%p)\n", This, iface, debugstr_guid(idPurpose), ppMediaStream);
152 for (i = 0; i < This->nbStreams; i++)
154 IMediaStream_GetInformation(This->pStreams[i], &PurposeId, NULL);
155 if (IsEqualIID(&PurposeId, idPurpose))
157 *ppMediaStream = This->pStreams[i];
158 IMediaStream_AddRef(*ppMediaStream);
163 return MS_E_NOSTREAM;
166 static HRESULT WINAPI IAMMultiMediaStreamImpl_EnumMediaStreams(IAMMultiMediaStream* iface, LONG Index, IMediaStream** ppMediaStream)
168 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
170 FIXME("(%p/%p)->(%d,%p) stub!\n", This, iface, Index, ppMediaStream);
175 static HRESULT WINAPI IAMMultiMediaStreamImpl_GetState(IAMMultiMediaStream* iface, STREAM_STATE* pCurrentState)
177 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
179 FIXME("(%p/%p)->(%p) stub!\n", This, iface, pCurrentState);
184 static HRESULT WINAPI IAMMultiMediaStreamImpl_SetState(IAMMultiMediaStream* iface, STREAM_STATE new_state)
186 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
187 HRESULT hr = E_INVALIDARG;
189 TRACE("(%p/%p)->(%u)\n", This, iface, new_state);
191 if (new_state == STREAMSTATE_RUN)
192 hr = IMediaControl_Run(This->media_control);
193 else if (new_state == STREAMSTATE_STOP)
194 hr = IMediaControl_Stop(This->media_control);
199 static HRESULT WINAPI IAMMultiMediaStreamImpl_GetTime(IAMMultiMediaStream* iface, STREAM_TIME* pCurrentTime)
201 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
203 FIXME("(%p/%p)->(%p) stub!\n", This, iface, pCurrentTime);
208 static HRESULT WINAPI IAMMultiMediaStreamImpl_GetDuration(IAMMultiMediaStream* iface, STREAM_TIME* pDuration)
210 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
212 FIXME("(%p/%p)->(%p) stub!\n", This, iface, pDuration);
217 static HRESULT WINAPI IAMMultiMediaStreamImpl_Seek(IAMMultiMediaStream* iface, STREAM_TIME seek_time)
219 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
221 TRACE("(%p/%p)->(%s)\n", This, iface, wine_dbgstr_longlong(seek_time));
223 return IMediaSeeking_SetPositions(This->media_seeking, &seek_time, AM_SEEKING_AbsolutePositioning, NULL, AM_SEEKING_NoPositioning);
226 static HRESULT WINAPI IAMMultiMediaStreamImpl_GetEndOfStream(IAMMultiMediaStream* iface, HANDLE* phEOS)
228 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
230 FIXME("(%p/%p)->(%p) stub!\n", This, iface, phEOS);
235 /*** IAMMultiMediaStream methods ***/
236 static HRESULT WINAPI IAMMultiMediaStreamImpl_Initialize(IAMMultiMediaStream* iface, STREAM_TYPE StreamType, DWORD dwFlags, IGraphBuilder* pFilterGraph)
238 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
240 const WCHAR filternameW[] = {'M','e','d','i','a','S','t','r','e','a','m','F','i','l','t','e','r',0};
242 TRACE("(%p/%p)->(%x,%x,%p)\n", This, iface, (DWORD)StreamType, dwFlags, pFilterGraph);
246 This->pFilterGraph = pFilterGraph;
247 IGraphBuilder_AddRef(This->pFilterGraph);
251 hr = CoCreateInstance(&CLSID_FilterGraph, NULL, CLSCTX_INPROC_SERVER, &IID_IGraphBuilder, (LPVOID*)&This->pFilterGraph);
256 This->StreamType = StreamType;
257 hr = IGraphBuilder_QueryInterface(This->pFilterGraph, &IID_IMediaSeeking, (void**)&This->media_seeking);
259 IGraphBuilder_QueryInterface(This->pFilterGraph, &IID_IMediaControl, (void**)&This->media_control);
261 hr = CoCreateInstance(&CLSID_MediaStreamFilter, NULL, CLSCTX_INPROC_SERVER, &IID_IBaseFilter, (LPVOID*)&This->media_stream_filter);
263 IGraphBuilder_AddFilter(This->pFilterGraph, This->media_stream_filter, filternameW);
266 IMediaEventEx* media_event = NULL;
267 hr = IGraphBuilder_QueryInterface(This->pFilterGraph, &IID_IMediaEventEx, (void**)&media_event);
269 hr = IMediaEventEx_GetEventHandle(media_event, &This->event);
271 hr = IMediaEventEx_SetNotifyFlags(media_event, AM_MEDIAEVENT_NONOTIFY);
273 IMediaEventEx_Release(media_event);
279 if (This->media_stream_filter)
280 IBaseFilter_Release(This->media_stream_filter);
281 This->media_stream_filter = NULL;
282 if (This->media_seeking)
283 IMediaSeeking_Release(This->media_seeking);
284 This->media_seeking = NULL;
285 if (This->media_control)
286 IMediaControl_Release(This->media_control);
287 This->media_control = NULL;
288 if (This->pFilterGraph)
289 IGraphBuilder_Release(This->pFilterGraph);
290 This->pFilterGraph = NULL;
296 static HRESULT WINAPI IAMMultiMediaStreamImpl_GetFilterGraph(IAMMultiMediaStream* iface, IGraphBuilder** ppGraphBuilder)
298 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
300 TRACE("(%p/%p)->(%p)\n", This, iface, ppGraphBuilder);
305 if (This->pFilterGraph)
306 return IGraphBuilder_QueryInterface(This->pFilterGraph, &IID_IGraphBuilder, (void**)ppGraphBuilder);
308 *ppGraphBuilder = NULL;
313 static HRESULT WINAPI IAMMultiMediaStreamImpl_GetFilter(IAMMultiMediaStream* iface, IMediaStreamFilter** ppFilter)
315 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
318 TRACE("(%p/%p)->(%p)\n", This, iface, ppFilter);
325 if (This->media_stream_filter)
326 hr = IBaseFilter_QueryInterface(This->media_stream_filter, &IID_IMediaStreamFilter, (LPVOID*)ppFilter);
331 static HRESULT WINAPI IAMMultiMediaStreamImpl_AddMediaStream(IAMMultiMediaStream* iface, IUnknown* stream_object, const MSPID* PurposeId,
332 DWORD dwFlags, IMediaStream** ppNewStream)
334 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
336 IMediaStream* pStream;
337 IMediaStream** pNewStreams;
339 TRACE("(%p/%p)->(%p,%s,%x,%p)\n", This, iface, stream_object, debugstr_guid(PurposeId), dwFlags, ppNewStream);
341 if (!IsEqualGUID(PurposeId, &MSPID_PrimaryVideo) && !IsEqualGUID(PurposeId, &MSPID_PrimaryAudio))
342 return MS_E_PURPOSEID;
345 FIXME("Specifying a stream object in params is not yet supported\n");
347 if (dwFlags & AMMSF_ADDDEFAULTRENDERER)
349 if (IsEqualGUID(PurposeId, &MSPID_PrimaryVideo))
351 /* Default renderer not supported by video stream */
352 return MS_E_PURPOSEID;
356 IBaseFilter* dsoundrender_filter;
358 /* Create the default renderer for audio */
359 hr = CoCreateInstance(&CLSID_DSoundRender, NULL, CLSCTX_INPROC_SERVER, &IID_IBaseFilter, (LPVOID*)&dsoundrender_filter);
362 hr = IGraphBuilder_AddFilter(This->pFilterGraph, dsoundrender_filter, NULL);
363 IBaseFilter_Release(dsoundrender_filter);
366 /* No media stream created when the default renderer is used */
371 if (IsEqualGUID(PurposeId, &MSPID_PrimaryVideo))
372 hr = ddrawmediastream_create((IMultiMediaStream*)iface, PurposeId, This->StreamType, &pStream);
374 hr = audiomediastream_create((IMultiMediaStream*)iface, PurposeId, This->StreamType, &pStream);
377 pNewStreams = CoTaskMemRealloc(This->pStreams, (This->nbStreams+1) * sizeof(IMediaStream*));
380 IMediaStream_Release(pStream);
381 return E_OUTOFMEMORY;
383 This->pStreams = pNewStreams;
384 This->pStreams[This->nbStreams] = pStream;
388 *ppNewStream = pStream;
393 /* Add stream to the media stream filter */
394 IMediaStreamFilter_AddMediaStream((IMediaStreamFilter*)This->media_stream_filter, (IAMMediaStream*)pStream);
400 static HRESULT WINAPI IAMMultiMediaStreamImpl_OpenFile(IAMMultiMediaStream* iface, LPCWSTR filename, DWORD flags)
402 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
404 IBaseFilter *BaseFilter = NULL;
405 IEnumPins *EnumPins = NULL;
407 PIN_DIRECTION pin_direction;
408 const WCHAR sourceW[] = {'S','o','u','r','c','e',0};
410 TRACE("(%p/%p)->(%s,%x)\n", This, iface, debugstr_w(filename), flags);
415 /* If Initialize was not called before, we do it here */
416 if (!This->pFilterGraph)
417 ret = IAMMultiMediaStream_Initialize(iface, STREAMTYPE_READ, 0, NULL);
420 ret = IGraphBuilder_AddSourceFilter(This->pFilterGraph, filename, sourceW, &BaseFilter);
423 ret = IBaseFilter_EnumPins(BaseFilter, &EnumPins);
426 ret = IEnumPins_Next(EnumPins, 1, &ipin, NULL);
430 ret = IPin_QueryDirection(ipin, &pin_direction);
431 if (ret == S_OK && pin_direction == PINDIR_OUTPUT)
435 if (SUCCEEDED(ret) && !(flags & AMMSF_NORENDER))
436 ret = IGraphBuilder_Render(This->pFilterGraph, This->ipin);
439 IEnumPins_Release(EnumPins);
441 IBaseFilter_Release(BaseFilter);
445 static HRESULT WINAPI IAMMultiMediaStreamImpl_OpenMoniker(IAMMultiMediaStream* iface, IBindCtx* pCtx, IMoniker* pMoniker, DWORD dwFlags)
447 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
449 FIXME("(%p/%p)->(%p,%p,%x) stub!\n", This, iface, pCtx, pMoniker, dwFlags);
454 static HRESULT WINAPI IAMMultiMediaStreamImpl_Render(IAMMultiMediaStream* iface, DWORD dwFlags)
456 IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
458 FIXME("(%p/%p)->(%x) partial stub!\n", This, iface, dwFlags);
460 if(dwFlags != AMMSF_NOCLOCK)
463 return IGraphBuilder_Render(This->pFilterGraph, This->ipin);
466 static const IAMMultiMediaStreamVtbl AM_Vtbl =
468 IAMMultiMediaStreamImpl_QueryInterface,
469 IAMMultiMediaStreamImpl_AddRef,
470 IAMMultiMediaStreamImpl_Release,
471 IAMMultiMediaStreamImpl_GetInformation,
472 IAMMultiMediaStreamImpl_GetMediaStream,
473 IAMMultiMediaStreamImpl_EnumMediaStreams,
474 IAMMultiMediaStreamImpl_GetState,
475 IAMMultiMediaStreamImpl_SetState,
476 IAMMultiMediaStreamImpl_GetTime,
477 IAMMultiMediaStreamImpl_GetDuration,
478 IAMMultiMediaStreamImpl_Seek,
479 IAMMultiMediaStreamImpl_GetEndOfStream,
480 IAMMultiMediaStreamImpl_Initialize,
481 IAMMultiMediaStreamImpl_GetFilterGraph,
482 IAMMultiMediaStreamImpl_GetFilter,
483 IAMMultiMediaStreamImpl_AddMediaStream,
484 IAMMultiMediaStreamImpl_OpenFile,
485 IAMMultiMediaStreamImpl_OpenMoniker,
486 IAMMultiMediaStreamImpl_Render