488 lines
16 KiB
C
488 lines
16 KiB
C
/*
|
|
* Implementation of IAMMultiMediaStream Interface
|
|
*
|
|
* Copyright 2004, 2012 Christian Costa
|
|
* Copyright 2006 Ivan Leo Puoti
|
|
*
|
|
* 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 "wine/debug.h"
|
|
|
|
#define COBJMACROS
|
|
|
|
#include "winbase.h"
|
|
#include "wingdi.h"
|
|
|
|
#include "amstream_private.h"
|
|
#include "amstream.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(amstream);
|
|
|
|
typedef struct {
|
|
IAMMultiMediaStream IAMMultiMediaStream_iface;
|
|
LONG ref;
|
|
IGraphBuilder* pFilterGraph;
|
|
IMediaSeeking* media_seeking;
|
|
IMediaControl* media_control;
|
|
IBaseFilter* media_stream_filter;
|
|
IPin* ipin;
|
|
ULONG nbStreams;
|
|
IMediaStream** pStreams;
|
|
STREAM_TYPE StreamType;
|
|
OAEVENT event;
|
|
} IAMMultiMediaStreamImpl;
|
|
|
|
static inline IAMMultiMediaStreamImpl *impl_from_IAMMultiMediaStream(IAMMultiMediaStream *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, IAMMultiMediaStreamImpl, IAMMultiMediaStream_iface);
|
|
}
|
|
|
|
static const struct IAMMultiMediaStreamVtbl AM_Vtbl;
|
|
|
|
HRESULT AM_create(IUnknown *pUnkOuter, LPVOID *ppObj)
|
|
{
|
|
IAMMultiMediaStreamImpl* object;
|
|
|
|
TRACE("(%p,%p)\n", pUnkOuter, ppObj);
|
|
|
|
if( pUnkOuter )
|
|
return CLASS_E_NOAGGREGATION;
|
|
|
|
object = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IAMMultiMediaStreamImpl));
|
|
if (!object)
|
|
return E_OUTOFMEMORY;
|
|
|
|
object->IAMMultiMediaStream_iface.lpVtbl = &AM_Vtbl;
|
|
object->ref = 1;
|
|
|
|
*ppObj = object;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
/*** IUnknown methods ***/
|
|
static HRESULT WINAPI IAMMultiMediaStreamImpl_QueryInterface(IAMMultiMediaStream* iface, REFIID riid, void** ppvObject)
|
|
{
|
|
IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
|
|
|
|
TRACE("(%p/%p)->(%s,%p)\n", iface, This, debugstr_guid(riid), ppvObject);
|
|
|
|
if (IsEqualGUID(riid, &IID_IUnknown) ||
|
|
IsEqualGUID(riid, &IID_IMultiMediaStream) ||
|
|
IsEqualGUID(riid, &IID_IAMMultiMediaStream))
|
|
{
|
|
IAMMultiMediaStream_AddRef(iface);
|
|
*ppvObject = iface;
|
|
return S_OK;
|
|
}
|
|
|
|
ERR("(%p)->(%s,%p),not found\n",This,debugstr_guid(riid),ppvObject);
|
|
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
static ULONG WINAPI IAMMultiMediaStreamImpl_AddRef(IAMMultiMediaStream* iface)
|
|
{
|
|
IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
|
|
|
|
TRACE("(%p/%p)\n", iface, This);
|
|
|
|
return InterlockedIncrement(&This->ref);
|
|
}
|
|
|
|
static ULONG WINAPI IAMMultiMediaStreamImpl_Release(IAMMultiMediaStream* iface)
|
|
{
|
|
IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
|
|
ULONG ref = InterlockedDecrement(&This->ref);
|
|
ULONG i;
|
|
|
|
TRACE("(%p/%p)\n", iface, This);
|
|
|
|
if (!ref)
|
|
{
|
|
for(i = 0; i < This->nbStreams; i++)
|
|
IMediaStream_Release(This->pStreams[i]);
|
|
if (This->ipin)
|
|
IPin_Release(This->ipin);
|
|
if (This->media_stream_filter)
|
|
IBaseFilter_Release(This->media_stream_filter);
|
|
if (This->media_seeking)
|
|
IMediaSeeking_Release(This->media_seeking);
|
|
if (This->media_control)
|
|
IMediaControl_Release(This->media_control);
|
|
if (This->pFilterGraph)
|
|
IGraphBuilder_Release(This->pFilterGraph);
|
|
HeapFree(GetProcessHeap(), 0, This);
|
|
}
|
|
|
|
return ref;
|
|
}
|
|
|
|
/*** IMultiMediaStream methods ***/
|
|
static HRESULT WINAPI IAMMultiMediaStreamImpl_GetInformation(IAMMultiMediaStream* iface, DWORD* pdwFlags, STREAM_TYPE* pStreamType)
|
|
{
|
|
IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
|
|
|
|
FIXME("(%p/%p)->(%p,%p) stub!\n", This, iface, pdwFlags, pStreamType);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI IAMMultiMediaStreamImpl_GetMediaStream(IAMMultiMediaStream* iface, REFMSPID idPurpose, IMediaStream** ppMediaStream)
|
|
{
|
|
IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
|
|
MSPID PurposeId;
|
|
unsigned int i;
|
|
|
|
TRACE("(%p/%p)->(%s,%p)\n", This, iface, debugstr_guid(idPurpose), ppMediaStream);
|
|
|
|
for (i = 0; i < This->nbStreams; i++)
|
|
{
|
|
IMediaStream_GetInformation(This->pStreams[i], &PurposeId, NULL);
|
|
if (IsEqualIID(&PurposeId, idPurpose))
|
|
{
|
|
*ppMediaStream = This->pStreams[i];
|
|
IMediaStream_AddRef(*ppMediaStream);
|
|
return S_OK;
|
|
}
|
|
}
|
|
|
|
return MS_E_NOSTREAM;
|
|
}
|
|
|
|
static HRESULT WINAPI IAMMultiMediaStreamImpl_EnumMediaStreams(IAMMultiMediaStream* iface, LONG Index, IMediaStream** ppMediaStream)
|
|
{
|
|
IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
|
|
|
|
FIXME("(%p/%p)->(%d,%p) stub!\n", This, iface, Index, ppMediaStream);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI IAMMultiMediaStreamImpl_GetState(IAMMultiMediaStream* iface, STREAM_STATE* pCurrentState)
|
|
{
|
|
IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
|
|
|
|
FIXME("(%p/%p)->(%p) stub!\n", This, iface, pCurrentState);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI IAMMultiMediaStreamImpl_SetState(IAMMultiMediaStream* iface, STREAM_STATE new_state)
|
|
{
|
|
IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
|
|
HRESULT hr = E_INVALIDARG;
|
|
|
|
TRACE("(%p/%p)->(%u)\n", This, iface, new_state);
|
|
|
|
if (new_state == STREAMSTATE_RUN)
|
|
hr = IMediaControl_Run(This->media_control);
|
|
else if (new_state == STREAMSTATE_STOP)
|
|
hr = IMediaControl_Stop(This->media_control);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI IAMMultiMediaStreamImpl_GetTime(IAMMultiMediaStream* iface, STREAM_TIME* pCurrentTime)
|
|
{
|
|
IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
|
|
|
|
FIXME("(%p/%p)->(%p) stub!\n", This, iface, pCurrentTime);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI IAMMultiMediaStreamImpl_GetDuration(IAMMultiMediaStream* iface, STREAM_TIME* pDuration)
|
|
{
|
|
IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
|
|
|
|
FIXME("(%p/%p)->(%p) stub!\n", This, iface, pDuration);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI IAMMultiMediaStreamImpl_Seek(IAMMultiMediaStream* iface, STREAM_TIME seek_time)
|
|
{
|
|
IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
|
|
|
|
TRACE("(%p/%p)->(%s)\n", This, iface, wine_dbgstr_longlong(seek_time));
|
|
|
|
return IMediaSeeking_SetPositions(This->media_seeking, &seek_time, AM_SEEKING_AbsolutePositioning, NULL, AM_SEEKING_NoPositioning);
|
|
}
|
|
|
|
static HRESULT WINAPI IAMMultiMediaStreamImpl_GetEndOfStream(IAMMultiMediaStream* iface, HANDLE* phEOS)
|
|
{
|
|
IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
|
|
|
|
FIXME("(%p/%p)->(%p) stub!\n", This, iface, phEOS);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
/*** IAMMultiMediaStream methods ***/
|
|
static HRESULT WINAPI IAMMultiMediaStreamImpl_Initialize(IAMMultiMediaStream* iface, STREAM_TYPE StreamType, DWORD dwFlags, IGraphBuilder* pFilterGraph)
|
|
{
|
|
IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
|
|
HRESULT hr = S_OK;
|
|
const WCHAR filternameW[] = {'M','e','d','i','a','S','t','r','e','a','m','F','i','l','t','e','r',0};
|
|
|
|
TRACE("(%p/%p)->(%x,%x,%p)\n", This, iface, (DWORD)StreamType, dwFlags, pFilterGraph);
|
|
|
|
if (pFilterGraph)
|
|
{
|
|
This->pFilterGraph = pFilterGraph;
|
|
IGraphBuilder_AddRef(This->pFilterGraph);
|
|
}
|
|
else
|
|
{
|
|
hr = CoCreateInstance(&CLSID_FilterGraph, NULL, CLSCTX_INPROC_SERVER, &IID_IGraphBuilder, (LPVOID*)&This->pFilterGraph);
|
|
}
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
This->StreamType = StreamType;
|
|
hr = IGraphBuilder_QueryInterface(This->pFilterGraph, &IID_IMediaSeeking, (void**)&This->media_seeking);
|
|
if (SUCCEEDED(hr))
|
|
hr = IGraphBuilder_QueryInterface(This->pFilterGraph, &IID_IMediaControl, (void**)&This->media_control);
|
|
if (SUCCEEDED(hr))
|
|
hr = CoCreateInstance(&CLSID_MediaStreamFilter, NULL, CLSCTX_INPROC_SERVER, &IID_IBaseFilter, (LPVOID*)&This->media_stream_filter);
|
|
if (SUCCEEDED(hr))
|
|
hr = IGraphBuilder_AddFilter(This->pFilterGraph, This->media_stream_filter, filternameW);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
IMediaEventEx* media_event = NULL;
|
|
hr = IGraphBuilder_QueryInterface(This->pFilterGraph, &IID_IMediaEventEx, (void**)&media_event);
|
|
if (SUCCEEDED(hr))
|
|
hr = IMediaEventEx_GetEventHandle(media_event, &This->event);
|
|
if (SUCCEEDED(hr))
|
|
hr = IMediaEventEx_SetNotifyFlags(media_event, AM_MEDIAEVENT_NONOTIFY);
|
|
if (media_event)
|
|
IMediaEventEx_Release(media_event);
|
|
}
|
|
}
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
if (This->media_stream_filter)
|
|
IBaseFilter_Release(This->media_stream_filter);
|
|
This->media_stream_filter = NULL;
|
|
if (This->media_seeking)
|
|
IMediaSeeking_Release(This->media_seeking);
|
|
This->media_seeking = NULL;
|
|
if (This->media_control)
|
|
IMediaControl_Release(This->media_control);
|
|
This->media_control = NULL;
|
|
if (This->pFilterGraph)
|
|
IGraphBuilder_Release(This->pFilterGraph);
|
|
This->pFilterGraph = NULL;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI IAMMultiMediaStreamImpl_GetFilterGraph(IAMMultiMediaStream* iface, IGraphBuilder** ppGraphBuilder)
|
|
{
|
|
IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
|
|
|
|
TRACE("(%p/%p)->(%p)\n", This, iface, ppGraphBuilder);
|
|
|
|
if (!ppGraphBuilder)
|
|
return E_POINTER;
|
|
|
|
if (This->pFilterGraph)
|
|
return IGraphBuilder_QueryInterface(This->pFilterGraph, &IID_IGraphBuilder, (void**)ppGraphBuilder);
|
|
else
|
|
*ppGraphBuilder = NULL;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI IAMMultiMediaStreamImpl_GetFilter(IAMMultiMediaStream* iface, IMediaStreamFilter** ppFilter)
|
|
{
|
|
IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
|
|
HRESULT hr = S_OK;
|
|
|
|
TRACE("(%p/%p)->(%p)\n", This, iface, ppFilter);
|
|
|
|
if (!ppFilter)
|
|
return E_POINTER;
|
|
|
|
*ppFilter = NULL;
|
|
|
|
if (This->media_stream_filter)
|
|
hr = IBaseFilter_QueryInterface(This->media_stream_filter, &IID_IMediaStreamFilter, (LPVOID*)ppFilter);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI IAMMultiMediaStreamImpl_AddMediaStream(IAMMultiMediaStream* iface, IUnknown* stream_object, const MSPID* PurposeId,
|
|
DWORD dwFlags, IMediaStream** ppNewStream)
|
|
{
|
|
IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
|
|
HRESULT hr;
|
|
IMediaStream* pStream;
|
|
IMediaStream** pNewStreams;
|
|
|
|
TRACE("(%p/%p)->(%p,%s,%x,%p)\n", This, iface, stream_object, debugstr_guid(PurposeId), dwFlags, ppNewStream);
|
|
|
|
if (!IsEqualGUID(PurposeId, &MSPID_PrimaryVideo) && !IsEqualGUID(PurposeId, &MSPID_PrimaryAudio))
|
|
return MS_E_PURPOSEID;
|
|
|
|
if (stream_object)
|
|
FIXME("Specifying a stream object in params is not yet supported\n");
|
|
|
|
if (dwFlags & AMMSF_ADDDEFAULTRENDERER)
|
|
{
|
|
if (IsEqualGUID(PurposeId, &MSPID_PrimaryVideo))
|
|
{
|
|
/* Default renderer not supported by video stream */
|
|
return MS_E_PURPOSEID;
|
|
}
|
|
else
|
|
{
|
|
IBaseFilter* dsoundrender_filter;
|
|
|
|
/* Create the default renderer for audio */
|
|
hr = CoCreateInstance(&CLSID_DSoundRender, NULL, CLSCTX_INPROC_SERVER, &IID_IBaseFilter, (LPVOID*)&dsoundrender_filter);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = IGraphBuilder_AddFilter(This->pFilterGraph, dsoundrender_filter, NULL);
|
|
IBaseFilter_Release(dsoundrender_filter);
|
|
}
|
|
|
|
/* No media stream created when the default renderer is used */
|
|
return hr;
|
|
}
|
|
}
|
|
|
|
if (IsEqualGUID(PurposeId, &MSPID_PrimaryVideo))
|
|
hr = ddrawmediastream_create((IMultiMediaStream*)iface, PurposeId, This->StreamType, &pStream);
|
|
else
|
|
hr = audiomediastream_create((IMultiMediaStream*)iface, PurposeId, This->StreamType, &pStream);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
pNewStreams = CoTaskMemRealloc(This->pStreams, (This->nbStreams+1) * sizeof(IMediaStream*));
|
|
if (!pNewStreams)
|
|
{
|
|
IMediaStream_Release(pStream);
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
This->pStreams = pNewStreams;
|
|
This->pStreams[This->nbStreams] = pStream;
|
|
This->nbStreams++;
|
|
|
|
if (ppNewStream)
|
|
*ppNewStream = pStream;
|
|
}
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
/* Add stream to the media stream filter */
|
|
IMediaStreamFilter_AddMediaStream((IMediaStreamFilter*)This->media_stream_filter, (IAMMediaStream*)pStream);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI IAMMultiMediaStreamImpl_OpenFile(IAMMultiMediaStream* iface, LPCWSTR filename, DWORD flags)
|
|
{
|
|
IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
|
|
HRESULT ret = S_OK;
|
|
IBaseFilter *BaseFilter = NULL;
|
|
IEnumPins *EnumPins = NULL;
|
|
IPin *ipin;
|
|
PIN_DIRECTION pin_direction;
|
|
const WCHAR sourceW[] = {'S','o','u','r','c','e',0};
|
|
|
|
TRACE("(%p/%p)->(%s,%x)\n", This, iface, debugstr_w(filename), flags);
|
|
|
|
if (!filename)
|
|
return E_POINTER;
|
|
|
|
/* If Initialize was not called before, we do it here */
|
|
if (!This->pFilterGraph)
|
|
ret = IAMMultiMediaStream_Initialize(iface, STREAMTYPE_READ, 0, NULL);
|
|
|
|
if (SUCCEEDED(ret))
|
|
ret = IGraphBuilder_AddSourceFilter(This->pFilterGraph, filename, sourceW, &BaseFilter);
|
|
|
|
if (SUCCEEDED(ret))
|
|
ret = IBaseFilter_EnumPins(BaseFilter, &EnumPins);
|
|
|
|
if (SUCCEEDED(ret))
|
|
ret = IEnumPins_Next(EnumPins, 1, &ipin, NULL);
|
|
|
|
if (SUCCEEDED(ret))
|
|
{
|
|
ret = IPin_QueryDirection(ipin, &pin_direction);
|
|
if (ret == S_OK && pin_direction == PINDIR_OUTPUT)
|
|
This->ipin = ipin;
|
|
}
|
|
|
|
if (SUCCEEDED(ret) && !(flags & AMMSF_NORENDER))
|
|
ret = IGraphBuilder_Render(This->pFilterGraph, This->ipin);
|
|
|
|
if (EnumPins)
|
|
IEnumPins_Release(EnumPins);
|
|
if (BaseFilter)
|
|
IBaseFilter_Release(BaseFilter);
|
|
return ret;
|
|
}
|
|
|
|
static HRESULT WINAPI IAMMultiMediaStreamImpl_OpenMoniker(IAMMultiMediaStream* iface, IBindCtx* pCtx, IMoniker* pMoniker, DWORD dwFlags)
|
|
{
|
|
IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
|
|
|
|
FIXME("(%p/%p)->(%p,%p,%x) stub!\n", This, iface, pCtx, pMoniker, dwFlags);
|
|
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI IAMMultiMediaStreamImpl_Render(IAMMultiMediaStream* iface, DWORD dwFlags)
|
|
{
|
|
IAMMultiMediaStreamImpl *This = impl_from_IAMMultiMediaStream(iface);
|
|
|
|
FIXME("(%p/%p)->(%x) partial stub!\n", This, iface, dwFlags);
|
|
|
|
if(dwFlags != AMMSF_NOCLOCK)
|
|
return E_INVALIDARG;
|
|
|
|
return IGraphBuilder_Render(This->pFilterGraph, This->ipin);
|
|
}
|
|
|
|
static const IAMMultiMediaStreamVtbl AM_Vtbl =
|
|
{
|
|
IAMMultiMediaStreamImpl_QueryInterface,
|
|
IAMMultiMediaStreamImpl_AddRef,
|
|
IAMMultiMediaStreamImpl_Release,
|
|
IAMMultiMediaStreamImpl_GetInformation,
|
|
IAMMultiMediaStreamImpl_GetMediaStream,
|
|
IAMMultiMediaStreamImpl_EnumMediaStreams,
|
|
IAMMultiMediaStreamImpl_GetState,
|
|
IAMMultiMediaStreamImpl_SetState,
|
|
IAMMultiMediaStreamImpl_GetTime,
|
|
IAMMultiMediaStreamImpl_GetDuration,
|
|
IAMMultiMediaStreamImpl_Seek,
|
|
IAMMultiMediaStreamImpl_GetEndOfStream,
|
|
IAMMultiMediaStreamImpl_Initialize,
|
|
IAMMultiMediaStreamImpl_GetFilterGraph,
|
|
IAMMultiMediaStreamImpl_GetFilter,
|
|
IAMMultiMediaStreamImpl_AddMediaStream,
|
|
IAMMultiMediaStreamImpl_OpenFile,
|
|
IAMMultiMediaStreamImpl_OpenMoniker,
|
|
IAMMultiMediaStreamImpl_Render
|
|
};
|