459 lines
15 KiB
C
459 lines
15 KiB
C
/*
|
|
* Copyright 2012 Austin English
|
|
*
|
|
* 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 "gst_private.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(wmvcore);
|
|
|
|
struct sync_reader
|
|
{
|
|
struct wm_reader reader;
|
|
|
|
IWMSyncReader2 IWMSyncReader2_iface;
|
|
|
|
WORD last_read_stream;
|
|
};
|
|
|
|
static struct sync_reader *impl_from_IWMSyncReader2(IWMSyncReader2 *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, struct sync_reader, IWMSyncReader2_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader_QueryInterface(IWMSyncReader2 *iface, REFIID iid, void **out)
|
|
{
|
|
struct sync_reader *reader = impl_from_IWMSyncReader2(iface);
|
|
|
|
return IWMProfile3_QueryInterface(&reader->reader.IWMProfile3_iface, iid, out);
|
|
}
|
|
|
|
static ULONG WINAPI WMSyncReader_AddRef(IWMSyncReader2 *iface)
|
|
{
|
|
struct sync_reader *reader = impl_from_IWMSyncReader2(iface);
|
|
|
|
return IWMProfile3_AddRef(&reader->reader.IWMProfile3_iface);
|
|
}
|
|
|
|
static ULONG WINAPI WMSyncReader_Release(IWMSyncReader2 *iface)
|
|
{
|
|
struct sync_reader *reader = impl_from_IWMSyncReader2(iface);
|
|
|
|
return IWMProfile3_Release(&reader->reader.IWMProfile3_iface);
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader_Close(IWMSyncReader2 *iface)
|
|
{
|
|
struct sync_reader *reader = impl_from_IWMSyncReader2(iface);
|
|
|
|
TRACE("reader %p.\n", reader);
|
|
|
|
return wm_reader_close(&reader->reader);
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader_GetMaxOutputSampleSize(IWMSyncReader2 *iface, DWORD output, DWORD *max)
|
|
{
|
|
struct sync_reader *This = impl_from_IWMSyncReader2(iface);
|
|
FIXME("(%p)->(%lu %p): stub!\n", This, output, max);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader_GetMaxStreamSampleSize(IWMSyncReader2 *iface, WORD stream, DWORD *max)
|
|
{
|
|
struct sync_reader *This = impl_from_IWMSyncReader2(iface);
|
|
FIXME("(%p)->(%d %p): stub!\n", This, stream, max);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader_GetNextSample(IWMSyncReader2 *iface,
|
|
WORD stream_number, INSSBuffer **sample, QWORD *pts, QWORD *duration,
|
|
DWORD *flags, DWORD *output_number, WORD *ret_stream_number)
|
|
{
|
|
struct sync_reader *reader = impl_from_IWMSyncReader2(iface);
|
|
HRESULT hr = NS_E_NO_MORE_SAMPLES;
|
|
struct wm_stream *stream;
|
|
WORD i;
|
|
|
|
TRACE("reader %p, stream_number %u, sample %p, pts %p, duration %p,"
|
|
" flags %p, output_number %p, ret_stream_number %p.\n",
|
|
reader, stream_number, sample, pts, duration, flags, output_number, ret_stream_number);
|
|
|
|
EnterCriticalSection(&reader->reader.cs);
|
|
|
|
if (!stream_number)
|
|
{
|
|
if (!output_number && !ret_stream_number)
|
|
{
|
|
LeaveCriticalSection(&reader->reader.cs);
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
for (i = 0; i < reader->reader.stream_count; ++i)
|
|
{
|
|
WORD index = (i + reader->last_read_stream + 1) % reader->reader.stream_count;
|
|
struct wm_stream *stream = &reader->reader.streams[index];
|
|
|
|
if (stream->selection == WMT_OFF)
|
|
continue;
|
|
|
|
hr = wm_reader_get_stream_sample(stream, sample, pts, duration, flags);
|
|
if (hr == S_OK)
|
|
{
|
|
if (output_number)
|
|
*output_number = index;
|
|
if (ret_stream_number)
|
|
*ret_stream_number = index + 1;
|
|
}
|
|
if (hr != NS_E_NO_MORE_SAMPLES)
|
|
{
|
|
reader->last_read_stream = index;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (!(stream = wm_reader_get_stream_by_stream_number(&reader->reader, stream_number)))
|
|
{
|
|
LeaveCriticalSection(&reader->reader.cs);
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
hr = wm_reader_get_stream_sample(stream, sample, pts, duration, flags);
|
|
if (hr == S_OK && output_number)
|
|
*output_number = stream->index;
|
|
if (ret_stream_number)
|
|
*ret_stream_number = stream->index + 1;
|
|
}
|
|
|
|
LeaveCriticalSection(&reader->reader.cs);
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader_GetOutputCount(IWMSyncReader2 *iface, DWORD *count)
|
|
{
|
|
struct sync_reader *reader = impl_from_IWMSyncReader2(iface);
|
|
|
|
TRACE("reader %p, count %p.\n", reader, count);
|
|
|
|
EnterCriticalSection(&reader->reader.cs);
|
|
*count = reader->reader.stream_count;
|
|
LeaveCriticalSection(&reader->reader.cs);
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader_GetOutputFormat(IWMSyncReader2 *iface,
|
|
DWORD output, DWORD index, IWMOutputMediaProps **props)
|
|
{
|
|
struct sync_reader *reader = impl_from_IWMSyncReader2(iface);
|
|
|
|
TRACE("reader %p, output %lu, index %lu, props %p.\n", reader, output, index, props);
|
|
|
|
return wm_reader_get_output_format(&reader->reader, output, index, props);
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader_GetOutputFormatCount(IWMSyncReader2 *iface, DWORD output, DWORD *count)
|
|
{
|
|
struct sync_reader *reader = impl_from_IWMSyncReader2(iface);
|
|
|
|
TRACE("reader %p, output %lu, count %p.\n", reader, output, count);
|
|
|
|
return wm_reader_get_output_format_count(&reader->reader, output, count);
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader_GetOutputNumberForStream(IWMSyncReader2 *iface,
|
|
WORD stream_number, DWORD *output)
|
|
{
|
|
struct sync_reader *reader = impl_from_IWMSyncReader2(iface);
|
|
|
|
TRACE("reader %p, stream_number %u, output %p.\n", reader, stream_number, output);
|
|
|
|
*output = stream_number - 1;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader_GetOutputProps(IWMSyncReader2 *iface,
|
|
DWORD output, IWMOutputMediaProps **props)
|
|
{
|
|
struct sync_reader *reader = impl_from_IWMSyncReader2(iface);
|
|
|
|
TRACE("reader %p, output %lu, props %p.\n", reader, output, props);
|
|
|
|
return wm_reader_get_output_props(&reader->reader, output, props);
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader_GetOutputSetting(IWMSyncReader2 *iface, DWORD output_num, const WCHAR *name,
|
|
WMT_ATTR_DATATYPE *type, BYTE *value, WORD *length)
|
|
{
|
|
struct sync_reader *This = impl_from_IWMSyncReader2(iface);
|
|
FIXME("(%p)->(%lu %s %p %p %p): stub!\n", This, output_num, debugstr_w(name), type, value, length);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader_GetReadStreamSamples(IWMSyncReader2 *iface, WORD stream_num, BOOL *compressed)
|
|
{
|
|
struct sync_reader *This = impl_from_IWMSyncReader2(iface);
|
|
FIXME("(%p)->(%d %p): stub!\n", This, stream_num, compressed);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader_GetStreamNumberForOutput(IWMSyncReader2 *iface,
|
|
DWORD output, WORD *stream_number)
|
|
{
|
|
struct sync_reader *reader = impl_from_IWMSyncReader2(iface);
|
|
|
|
TRACE("reader %p, output %lu, stream_number %p.\n", reader, output, stream_number);
|
|
|
|
*stream_number = output + 1;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader_GetStreamSelected(IWMSyncReader2 *iface,
|
|
WORD stream_number, WMT_STREAM_SELECTION *selection)
|
|
{
|
|
struct sync_reader *reader = impl_from_IWMSyncReader2(iface);
|
|
|
|
TRACE("reader %p, stream_number %u, selection %p.\n", reader, stream_number, selection);
|
|
|
|
return wm_reader_get_stream_selection(&reader->reader, stream_number, selection);
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader_Open(IWMSyncReader2 *iface, const WCHAR *filename)
|
|
{
|
|
struct sync_reader *reader = impl_from_IWMSyncReader2(iface);
|
|
|
|
TRACE("reader %p, filename %s.\n", reader, debugstr_w(filename));
|
|
|
|
return wm_reader_open_file(&reader->reader, filename);
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader_OpenStream(IWMSyncReader2 *iface, IStream *stream)
|
|
{
|
|
struct sync_reader *reader = impl_from_IWMSyncReader2(iface);
|
|
|
|
TRACE("reader %p, stream %p.\n", reader, stream);
|
|
|
|
return wm_reader_open_stream(&reader->reader, stream);
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader_SetOutputProps(IWMSyncReader2 *iface, DWORD output, IWMOutputMediaProps *props)
|
|
{
|
|
struct sync_reader *reader = impl_from_IWMSyncReader2(iface);
|
|
|
|
TRACE("reader %p, output %lu, props %p.\n", reader, output, props);
|
|
|
|
return wm_reader_set_output_props(&reader->reader, output, props);
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader_SetOutputSetting(IWMSyncReader2 *iface, DWORD output,
|
|
const WCHAR *name, WMT_ATTR_DATATYPE type, const BYTE *value, WORD size)
|
|
{
|
|
struct sync_reader *reader = impl_from_IWMSyncReader2(iface);
|
|
|
|
TRACE("reader %p, output %lu, name %s, type %#x, value %p, size %u.\n",
|
|
reader, output, debugstr_w(name), type, value, size);
|
|
|
|
if (!wcscmp(name, L"VideoSampleDurations"))
|
|
{
|
|
FIXME("Ignoring VideoSampleDurations setting.\n");
|
|
return S_OK;
|
|
}
|
|
if (!wcscmp(name, L"EnableDiscreteOutput"))
|
|
{
|
|
FIXME("Ignoring EnableDiscreteOutput setting.\n");
|
|
return S_OK;
|
|
}
|
|
if (!wcscmp(name, L"SpeakerConfig"))
|
|
{
|
|
FIXME("Ignoring SpeakerConfig setting.\n");
|
|
return S_OK;
|
|
}
|
|
else
|
|
{
|
|
FIXME("Unknown setting %s; returning E_NOTIMPL.\n", debugstr_w(name));
|
|
return E_NOTIMPL;
|
|
}
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader_SetRange(IWMSyncReader2 *iface, QWORD start, LONGLONG duration)
|
|
{
|
|
struct sync_reader *reader = impl_from_IWMSyncReader2(iface);
|
|
|
|
TRACE("reader %p, start %I64u, duration %I64d.\n", reader, start, duration);
|
|
|
|
wm_reader_seek(&reader->reader, start, duration);
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader_SetRangeByFrame(IWMSyncReader2 *iface, WORD stream_num, QWORD frame_num,
|
|
LONGLONG frames)
|
|
{
|
|
struct sync_reader *This = impl_from_IWMSyncReader2(iface);
|
|
FIXME("(%p)->(%d %s %s): stub!\n", This, stream_num, wine_dbgstr_longlong(frame_num), wine_dbgstr_longlong(frames));
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader_SetReadStreamSamples(IWMSyncReader2 *iface, WORD stream_number, BOOL compressed)
|
|
{
|
|
struct sync_reader *reader = impl_from_IWMSyncReader2(iface);
|
|
|
|
TRACE("reader %p, stream_index %u, compressed %d.\n", reader, stream_number, compressed);
|
|
|
|
return wm_reader_set_read_compressed(&reader->reader, stream_number, compressed);
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader_SetStreamsSelected(IWMSyncReader2 *iface,
|
|
WORD count, WORD *stream_numbers, WMT_STREAM_SELECTION *selections)
|
|
{
|
|
struct sync_reader *reader = impl_from_IWMSyncReader2(iface);
|
|
|
|
TRACE("reader %p, count %u, stream_numbers %p, selections %p.\n",
|
|
reader, count, stream_numbers, selections);
|
|
|
|
return wm_reader_set_streams_selected(&reader->reader, count, stream_numbers, selections);
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader2_SetRangeByTimecode(IWMSyncReader2 *iface, WORD stream_num,
|
|
WMT_TIMECODE_EXTENSION_DATA *start, WMT_TIMECODE_EXTENSION_DATA *end)
|
|
{
|
|
struct sync_reader *This = impl_from_IWMSyncReader2(iface);
|
|
FIXME("(%p)->(%u %p %p): stub!\n", This, stream_num, start, end);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader2_SetRangeByFrameEx(IWMSyncReader2 *iface, WORD stream_num, QWORD frame_num,
|
|
LONGLONG frames_to_read, QWORD *starttime)
|
|
{
|
|
struct sync_reader *This = impl_from_IWMSyncReader2(iface);
|
|
FIXME("(%p)->(%u %s %s %p): stub!\n", This, stream_num, wine_dbgstr_longlong(frame_num),
|
|
wine_dbgstr_longlong(frames_to_read), starttime);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader2_SetAllocateForOutput(IWMSyncReader2 *iface, DWORD output_num, IWMReaderAllocatorEx *allocator)
|
|
{
|
|
struct sync_reader *This = impl_from_IWMSyncReader2(iface);
|
|
FIXME("(%p)->(%lu %p): stub!\n", This, output_num, allocator);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader2_GetAllocateForOutput(IWMSyncReader2 *iface, DWORD output_num, IWMReaderAllocatorEx **allocator)
|
|
{
|
|
struct sync_reader *This = impl_from_IWMSyncReader2(iface);
|
|
FIXME("(%p)->(%lu %p): stub!\n", This, output_num, allocator);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader2_SetAllocateForStream(IWMSyncReader2 *iface, DWORD stream_num, IWMReaderAllocatorEx *allocator)
|
|
{
|
|
struct sync_reader *This = impl_from_IWMSyncReader2(iface);
|
|
FIXME("(%p)->(%lu %p): stub!\n", This, stream_num, allocator);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static HRESULT WINAPI WMSyncReader2_GetAllocateForStream(IWMSyncReader2 *iface, DWORD stream_num, IWMReaderAllocatorEx **allocator)
|
|
{
|
|
struct sync_reader *This = impl_from_IWMSyncReader2(iface);
|
|
FIXME("(%p)->(%lu %p): stub!\n", This, stream_num, allocator);
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
static const IWMSyncReader2Vtbl WMSyncReader2Vtbl = {
|
|
WMSyncReader_QueryInterface,
|
|
WMSyncReader_AddRef,
|
|
WMSyncReader_Release,
|
|
WMSyncReader_Open,
|
|
WMSyncReader_Close,
|
|
WMSyncReader_SetRange,
|
|
WMSyncReader_SetRangeByFrame,
|
|
WMSyncReader_GetNextSample,
|
|
WMSyncReader_SetStreamsSelected,
|
|
WMSyncReader_GetStreamSelected,
|
|
WMSyncReader_SetReadStreamSamples,
|
|
WMSyncReader_GetReadStreamSamples,
|
|
WMSyncReader_GetOutputSetting,
|
|
WMSyncReader_SetOutputSetting,
|
|
WMSyncReader_GetOutputCount,
|
|
WMSyncReader_GetOutputProps,
|
|
WMSyncReader_SetOutputProps,
|
|
WMSyncReader_GetOutputFormatCount,
|
|
WMSyncReader_GetOutputFormat,
|
|
WMSyncReader_GetOutputNumberForStream,
|
|
WMSyncReader_GetStreamNumberForOutput,
|
|
WMSyncReader_GetMaxOutputSampleSize,
|
|
WMSyncReader_GetMaxStreamSampleSize,
|
|
WMSyncReader_OpenStream,
|
|
WMSyncReader2_SetRangeByTimecode,
|
|
WMSyncReader2_SetRangeByFrameEx,
|
|
WMSyncReader2_SetAllocateForOutput,
|
|
WMSyncReader2_GetAllocateForOutput,
|
|
WMSyncReader2_SetAllocateForStream,
|
|
WMSyncReader2_GetAllocateForStream
|
|
};
|
|
|
|
static struct sync_reader *impl_from_wm_reader(struct wm_reader *iface)
|
|
{
|
|
return CONTAINING_RECORD(iface, struct sync_reader, reader);
|
|
}
|
|
|
|
static void *sync_reader_query_interface(struct wm_reader *iface, REFIID iid)
|
|
{
|
|
struct sync_reader *reader = impl_from_wm_reader(iface);
|
|
|
|
TRACE("reader %p, iid %s.\n", reader, debugstr_guid(iid));
|
|
|
|
if (IsEqualIID(iid, &IID_IWMSyncReader)
|
|
|| IsEqualIID(iid, &IID_IWMSyncReader2))
|
|
return &reader->IWMSyncReader2_iface;
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static void sync_reader_destroy(struct wm_reader *iface)
|
|
{
|
|
struct sync_reader *reader = impl_from_wm_reader(iface);
|
|
|
|
TRACE("reader %p.\n", reader);
|
|
|
|
wm_reader_close(&reader->reader);
|
|
wm_reader_cleanup(&reader->reader);
|
|
free(reader);
|
|
}
|
|
|
|
static const struct wm_reader_ops sync_reader_ops =
|
|
{
|
|
.query_interface = sync_reader_query_interface,
|
|
.destroy = sync_reader_destroy,
|
|
};
|
|
|
|
HRESULT WINAPI winegstreamer_create_wm_sync_reader(IWMSyncReader **reader)
|
|
{
|
|
struct sync_reader *object;
|
|
|
|
TRACE("reader %p.\n", reader);
|
|
|
|
if (!(object = calloc(1, sizeof(*object))))
|
|
return E_OUTOFMEMORY;
|
|
|
|
wm_reader_init(&object->reader, &sync_reader_ops);
|
|
|
|
object->IWMSyncReader2_iface.lpVtbl = &WMSyncReader2Vtbl;
|
|
|
|
TRACE("Created sync reader %p.\n", object);
|
|
*reader = (IWMSyncReader *)&object->IWMSyncReader2_iface;
|
|
return S_OK;
|
|
}
|