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"
31 WINE_DEFAULT_DEBUG_CHANNEL(amstream
);
34 IAMMultiMediaStream IAMMultiMediaStream_iface
;
36 IGraphBuilder
* pFilterGraph
;
37 IMediaSeeking
* media_seeking
;
38 IMediaControl
* media_control
;
39 IMediaStreamFilter
*media_stream_filter
;
42 IAMMediaStream
**pStreams
;
43 STREAM_TYPE StreamType
;
45 } IAMMultiMediaStreamImpl
;
47 static inline IAMMultiMediaStreamImpl
*impl_from_IAMMultiMediaStream(IAMMultiMediaStream
*iface
)
49 return CONTAINING_RECORD(iface
, IAMMultiMediaStreamImpl
, IAMMultiMediaStream_iface
);
52 static const struct IAMMultiMediaStreamVtbl AM_Vtbl
;
54 HRESULT
AM_create(IUnknown
*pUnkOuter
, LPVOID
*ppObj
)
56 IAMMultiMediaStreamImpl
* object
;
58 TRACE("(%p,%p)\n", pUnkOuter
, ppObj
);
61 return CLASS_E_NOAGGREGATION
;
63 object
= HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY
, sizeof(IAMMultiMediaStreamImpl
));
67 object
->IAMMultiMediaStream_iface
.lpVtbl
= &AM_Vtbl
;
70 *ppObj
= &object
->IAMMultiMediaStream_iface
;
75 /*** IUnknown methods ***/
76 static HRESULT WINAPI
IAMMultiMediaStreamImpl_QueryInterface(IAMMultiMediaStream
* iface
, REFIID riid
, void** ppvObject
)
78 IAMMultiMediaStreamImpl
*This
= impl_from_IAMMultiMediaStream(iface
);
80 TRACE("(%p/%p)->(%s,%p)\n", iface
, This
, debugstr_guid(riid
), ppvObject
);
82 if (IsEqualGUID(riid
, &IID_IUnknown
) ||
83 IsEqualGUID(riid
, &IID_IMultiMediaStream
) ||
84 IsEqualGUID(riid
, &IID_IAMMultiMediaStream
))
86 IAMMultiMediaStream_AddRef(iface
);
91 ERR("(%p)->(%s,%p),not found\n",This
,debugstr_guid(riid
),ppvObject
);
96 static ULONG WINAPI
IAMMultiMediaStreamImpl_AddRef(IAMMultiMediaStream
* iface
)
98 IAMMultiMediaStreamImpl
*This
= impl_from_IAMMultiMediaStream(iface
);
100 TRACE("(%p/%p)\n", iface
, This
);
102 return InterlockedIncrement(&This
->ref
);
105 static ULONG WINAPI
IAMMultiMediaStreamImpl_Release(IAMMultiMediaStream
* iface
)
107 IAMMultiMediaStreamImpl
*This
= impl_from_IAMMultiMediaStream(iface
);
108 ULONG ref
= InterlockedDecrement(&This
->ref
);
111 TRACE("(%p/%p)\n", iface
, This
);
115 for(i
= 0; i
< This
->nbStreams
; i
++)
116 IAMMediaStream_Release(This
->pStreams
[i
]);
117 CoTaskMemFree(This
->pStreams
);
119 IPin_Release(This
->ipin
);
120 if (This
->media_stream_filter
)
121 IMediaStreamFilter_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 IAMMediaStream_GetInformation(This
->pStreams
[i
], &PurposeId
, NULL
);
155 if (IsEqualIID(&PurposeId
, idPurpose
))
157 *ppMediaStream
= (IMediaStream
*)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 hr
= IGraphBuilder_QueryInterface(This
->pFilterGraph
, &IID_IMediaControl
, (void**)&This
->media_control
);
261 hr
= CoCreateInstance(&CLSID_MediaStreamFilter
, NULL
, CLSCTX_INPROC_SERVER
, &IID_IMediaStreamFilter
, (void**)&This
->media_stream_filter
);
263 hr
= IGraphBuilder_AddFilter(This
->pFilterGraph
, (IBaseFilter
*)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 IMediaStreamFilter_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
);
317 TRACE("(%p/%p)->(%p)\n", This
, iface
, ppFilter
);
322 *ppFilter
= This
->media_stream_filter
;
324 IMediaStreamFilter_AddRef(*ppFilter
);
329 static HRESULT WINAPI
IAMMultiMediaStreamImpl_AddMediaStream(IAMMultiMediaStream
* iface
, IUnknown
* stream_object
, const MSPID
* PurposeId
,
330 DWORD dwFlags
, IMediaStream
** ppNewStream
)
332 IAMMultiMediaStreamImpl
*This
= impl_from_IAMMultiMediaStream(iface
);
334 IAMMediaStream
* pStream
;
335 IAMMediaStream
** pNewStreams
;
337 TRACE("(%p/%p)->(%p,%s,%x,%p)\n", This
, iface
, stream_object
, debugstr_guid(PurposeId
), dwFlags
, ppNewStream
);
339 if (!IsEqualGUID(PurposeId
, &MSPID_PrimaryVideo
) && !IsEqualGUID(PurposeId
, &MSPID_PrimaryAudio
))
340 return MS_E_PURPOSEID
;
343 FIXME("Specifying a stream object in params is not yet supported\n");
345 if (dwFlags
& AMMSF_ADDDEFAULTRENDERER
)
347 if (IsEqualGUID(PurposeId
, &MSPID_PrimaryVideo
))
349 /* Default renderer not supported by video stream */
350 return MS_E_PURPOSEID
;
354 IBaseFilter
* dsoundrender_filter
;
356 /* Create the default renderer for audio */
357 hr
= CoCreateInstance(&CLSID_DSoundRender
, NULL
, CLSCTX_INPROC_SERVER
, &IID_IBaseFilter
, (LPVOID
*)&dsoundrender_filter
);
360 hr
= IGraphBuilder_AddFilter(This
->pFilterGraph
, dsoundrender_filter
, NULL
);
361 IBaseFilter_Release(dsoundrender_filter
);
364 /* No media stream created when the default renderer is used */
369 if (IsEqualGUID(PurposeId
, &MSPID_PrimaryVideo
))
370 hr
= ddrawmediastream_create((IMultiMediaStream
*)iface
, PurposeId
, This
->StreamType
, &pStream
);
372 hr
= audiomediastream_create((IMultiMediaStream
*)iface
, PurposeId
, This
->StreamType
, &pStream
);
375 pNewStreams
= CoTaskMemRealloc(This
->pStreams
, (This
->nbStreams
+1) * sizeof(IAMMediaStream
*));
378 IAMMediaStream_Release(pStream
);
379 return E_OUTOFMEMORY
;
381 This
->pStreams
= pNewStreams
;
382 This
->pStreams
[This
->nbStreams
] = pStream
;
386 *ppNewStream
= (IMediaStream
*)pStream
;
391 /* Add stream to the media stream filter */
392 IMediaStreamFilter_AddMediaStream(This
->media_stream_filter
, pStream
);
398 static HRESULT WINAPI
IAMMultiMediaStreamImpl_OpenFile(IAMMultiMediaStream
* iface
, LPCWSTR filename
, DWORD flags
)
400 IAMMultiMediaStreamImpl
*This
= impl_from_IAMMultiMediaStream(iface
);
402 IBaseFilter
*BaseFilter
= NULL
;
403 IEnumPins
*EnumPins
= NULL
;
405 PIN_DIRECTION pin_direction
;
406 const WCHAR sourceW
[] = {'S','o','u','r','c','e',0};
408 TRACE("(%p/%p)->(%s,%x)\n", This
, iface
, debugstr_w(filename
), flags
);
413 /* If Initialize was not called before, we do it here */
414 if (!This
->pFilterGraph
)
415 ret
= IAMMultiMediaStream_Initialize(iface
, STREAMTYPE_READ
, 0, NULL
);
418 ret
= IGraphBuilder_AddSourceFilter(This
->pFilterGraph
, filename
, sourceW
, &BaseFilter
);
421 ret
= IBaseFilter_EnumPins(BaseFilter
, &EnumPins
);
424 ret
= IEnumPins_Next(EnumPins
, 1, &ipin
, NULL
);
428 ret
= IPin_QueryDirection(ipin
, &pin_direction
);
429 if (ret
== S_OK
&& pin_direction
== PINDIR_OUTPUT
)
433 if (SUCCEEDED(ret
) && !(flags
& AMMSF_NORENDER
))
434 ret
= IGraphBuilder_Render(This
->pFilterGraph
, This
->ipin
);
437 IEnumPins_Release(EnumPins
);
439 IBaseFilter_Release(BaseFilter
);
443 static HRESULT WINAPI
IAMMultiMediaStreamImpl_OpenMoniker(IAMMultiMediaStream
* iface
, IBindCtx
* pCtx
, IMoniker
* pMoniker
, DWORD dwFlags
)
445 IAMMultiMediaStreamImpl
*This
= impl_from_IAMMultiMediaStream(iface
);
447 FIXME("(%p/%p)->(%p,%p,%x) stub!\n", This
, iface
, pCtx
, pMoniker
, dwFlags
);
452 static HRESULT WINAPI
IAMMultiMediaStreamImpl_Render(IAMMultiMediaStream
* iface
, DWORD dwFlags
)
454 IAMMultiMediaStreamImpl
*This
= impl_from_IAMMultiMediaStream(iface
);
456 FIXME("(%p/%p)->(%x) partial stub!\n", This
, iface
, dwFlags
);
458 if(dwFlags
!= AMMSF_NOCLOCK
)
461 return IGraphBuilder_Render(This
->pFilterGraph
, This
->ipin
);
464 static const IAMMultiMediaStreamVtbl AM_Vtbl
=
466 IAMMultiMediaStreamImpl_QueryInterface
,
467 IAMMultiMediaStreamImpl_AddRef
,
468 IAMMultiMediaStreamImpl_Release
,
469 IAMMultiMediaStreamImpl_GetInformation
,
470 IAMMultiMediaStreamImpl_GetMediaStream
,
471 IAMMultiMediaStreamImpl_EnumMediaStreams
,
472 IAMMultiMediaStreamImpl_GetState
,
473 IAMMultiMediaStreamImpl_SetState
,
474 IAMMultiMediaStreamImpl_GetTime
,
475 IAMMultiMediaStreamImpl_GetDuration
,
476 IAMMultiMediaStreamImpl_Seek
,
477 IAMMultiMediaStreamImpl_GetEndOfStream
,
478 IAMMultiMediaStreamImpl_Initialize
,
479 IAMMultiMediaStreamImpl_GetFilterGraph
,
480 IAMMultiMediaStreamImpl_GetFilter
,
481 IAMMultiMediaStreamImpl_AddMediaStream
,
482 IAMMultiMediaStreamImpl_OpenFile
,
483 IAMMultiMediaStreamImpl_OpenMoniker
,
484 IAMMultiMediaStreamImpl_Render