2 * GStreamer splitter + decoder, adapted from parser.c
4 * Copyright 2010 Maarten Lankhorst for CodeWeavers
5 * Copyright 2010 Aric Stewart for CodeWeavers
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
23 #include "gst_private.h"
24 #include "gst_guids.h"
30 #include "wine/unicode.h"
31 #include "wine/debug.h"
39 #include "wmcodecdsp.h"
42 WINE_DEFAULT_DEBUG_CHANNEL(gstreamer
);
44 static const GUID MEDIASUBTYPE_CVID
= {mmioFOURCC('c','v','i','d'), 0x0000, 0x0010, {0x80, 0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71}};
48 struct strmbase_filter filter
;
49 IAMStreamSelect IAMStreamSelect_iface
;
51 struct strmbase_sink sink
;
54 struct gstdemux_source
**sources
;
55 unsigned int source_count
;
60 BOOL initial
, ignore_flush
;
61 GstElement
*container
;
62 GstPad
*my_src
, *their_sink
;
64 guint64 start
, nextofs
, nextpullofs
, stop
;
65 HANDLE no_more_pads_event
, duration_event
, error_event
;
69 BOOL (*init_gst
)(struct gstdemux
*filter
);
70 HRESULT (*source_query_accept
)(struct gstdemux_source
*pin
, const AM_MEDIA_TYPE
*mt
);
71 HRESULT (*source_get_media_type
)(struct gstdemux_source
*pin
, unsigned int index
, AM_MEDIA_TYPE
*mt
);
74 struct gstdemux_source
76 struct strmbase_source pin
;
77 IQualityControl IQualityControl_iface
;
79 GstPad
*their_src
, *post_sink
, *post_src
, *my_sink
;
82 HANDLE caps_event
, eos_event
;
87 static inline struct gstdemux
*impl_from_strmbase_filter(struct strmbase_filter
*iface
)
89 return CONTAINING_RECORD(iface
, struct gstdemux
, filter
);
92 static const WCHAR wcsInputPinName
[] = {'i','n','p','u','t',' ','p','i','n',0};
93 static const IMediaSeekingVtbl GST_Seeking_Vtbl
;
94 static const IQualityControlVtbl GSTOutPin_QualityControl_Vtbl
;
96 static struct gstdemux_source
*create_pin(struct gstdemux
*filter
, const WCHAR
*name
);
97 static HRESULT
GST_RemoveOutputPins(struct gstdemux
*This
);
98 static HRESULT WINAPI
GST_ChangeCurrent(IMediaSeeking
*iface
);
99 static HRESULT WINAPI
GST_ChangeStop(IMediaSeeking
*iface
);
100 static HRESULT WINAPI
GST_ChangeRate(IMediaSeeking
*iface
);
102 static gboolean
amt_from_gst_caps_audio_raw(const GstCaps
*caps
, AM_MEDIA_TYPE
*amt
)
104 WAVEFORMATEXTENSIBLE
*wfe
;
109 if (!gst_audio_info_from_caps (&ainfo
, caps
))
112 wfe
= CoTaskMemAlloc(sizeof(*wfe
));
113 wfx
= (WAVEFORMATEX
*)wfe
;
114 amt
->majortype
= MEDIATYPE_Audio
;
115 amt
->subtype
= MEDIASUBTYPE_PCM
;
116 amt
->formattype
= FORMAT_WaveFormatEx
;
117 amt
->pbFormat
= (BYTE
*)wfe
;
118 amt
->cbFormat
= sizeof(*wfe
);
119 amt
->bFixedSizeSamples
= TRUE
;
120 amt
->bTemporalCompression
= FALSE
;
123 wfx
->wFormatTag
= WAVE_FORMAT_EXTENSIBLE
;
125 wfx
->nChannels
= ainfo
.channels
;
126 wfx
->nSamplesPerSec
= ainfo
.rate
;
127 depth
= GST_AUDIO_INFO_WIDTH(&ainfo
);
128 bpp
= GST_AUDIO_INFO_DEPTH(&ainfo
);
130 if (!depth
|| depth
> 32 || depth
% 8)
134 wfe
->Samples
.wValidBitsPerSample
= depth
;
135 wfx
->wBitsPerSample
= bpp
;
136 wfx
->cbSize
= sizeof(*wfe
)-sizeof(*wfx
);
137 switch (wfx
->nChannels
) {
138 case 1: wfe
->dwChannelMask
= KSAUDIO_SPEAKER_MONO
; break;
139 case 2: wfe
->dwChannelMask
= KSAUDIO_SPEAKER_STEREO
; break;
140 case 4: wfe
->dwChannelMask
= KSAUDIO_SPEAKER_SURROUND
; break;
141 case 5: wfe
->dwChannelMask
= (KSAUDIO_SPEAKER_5POINT1
& ~SPEAKER_LOW_FREQUENCY
); break;
142 case 6: wfe
->dwChannelMask
= KSAUDIO_SPEAKER_5POINT1
; break;
143 case 8: wfe
->dwChannelMask
= KSAUDIO_SPEAKER_7POINT1
; break;
145 wfe
->dwChannelMask
= 0;
147 if (GST_AUDIO_INFO_IS_FLOAT(&ainfo
)) {
148 amt
->subtype
= MEDIASUBTYPE_IEEE_FLOAT
;
149 wfe
->SubFormat
= KSDATAFORMAT_SUBTYPE_IEEE_FLOAT
;
151 wfe
->SubFormat
= KSDATAFORMAT_SUBTYPE_PCM
;
152 if (wfx
->nChannels
<= 2 && bpp
<= 16 && depth
== bpp
) {
153 wfx
->wFormatTag
= WAVE_FORMAT_PCM
;
155 amt
->cbFormat
= sizeof(WAVEFORMATEX
);
158 amt
->lSampleSize
= wfx
->nBlockAlign
= wfx
->nChannels
* wfx
->wBitsPerSample
/8;
159 wfx
->nAvgBytesPerSec
= wfx
->nSamplesPerSec
* wfx
->nBlockAlign
;
163 static gboolean
amt_from_gst_video_info(const GstVideoInfo
*info
, AM_MEDIA_TYPE
*amt
)
165 VIDEOINFOHEADER
*vih
;
166 BITMAPINFOHEADER
*bih
;
167 gint32 width
, height
;
169 width
= GST_VIDEO_INFO_WIDTH(info
);
170 height
= GST_VIDEO_INFO_HEIGHT(info
);
172 vih
= CoTaskMemAlloc(sizeof(*vih
));
173 bih
= &vih
->bmiHeader
;
175 amt
->formattype
= FORMAT_VideoInfo
;
176 amt
->pbFormat
= (BYTE
*)vih
;
177 amt
->cbFormat
= sizeof(*vih
);
178 amt
->bFixedSizeSamples
= FALSE
;
179 amt
->bTemporalCompression
= TRUE
;
180 amt
->lSampleSize
= 1;
182 ZeroMemory(vih
, sizeof(*vih
));
183 amt
->majortype
= MEDIATYPE_Video
;
185 if (GST_VIDEO_INFO_IS_RGB(info
))
187 switch (GST_VIDEO_INFO_FORMAT(info
))
189 case GST_VIDEO_FORMAT_BGRA
:
190 amt
->subtype
= MEDIASUBTYPE_ARGB32
;
191 bih
->biBitCount
= 32;
193 case GST_VIDEO_FORMAT_BGRx
:
194 amt
->subtype
= MEDIASUBTYPE_RGB32
;
195 bih
->biBitCount
= 32;
197 case GST_VIDEO_FORMAT_BGR
:
198 amt
->subtype
= MEDIASUBTYPE_RGB24
;
199 bih
->biBitCount
= 24;
201 case GST_VIDEO_FORMAT_BGR16
:
202 amt
->subtype
= MEDIASUBTYPE_RGB565
;
203 bih
->biBitCount
= 16;
205 case GST_VIDEO_FORMAT_BGR15
:
206 amt
->subtype
= MEDIASUBTYPE_RGB555
;
207 bih
->biBitCount
= 16;
210 FIXME("Unhandled type %s.\n", GST_VIDEO_INFO_NAME(info
));
214 bih
->biCompression
= BI_RGB
;
216 amt
->subtype
= MEDIATYPE_Video
;
217 if (!(amt
->subtype
.Data1
= gst_video_format_to_fourcc(GST_VIDEO_INFO_FORMAT(info
))))
222 switch (amt
->subtype
.Data1
) {
223 case mmioFOURCC('I','4','2','0'):
224 case mmioFOURCC('Y','V','1','2'):
225 case mmioFOURCC('N','V','1','2'):
226 case mmioFOURCC('N','V','2','1'):
227 bih
->biBitCount
= 12; break;
228 case mmioFOURCC('Y','U','Y','2'):
229 case mmioFOURCC('Y','V','Y','U'):
230 case mmioFOURCC('U','Y','V','Y'):
231 bih
->biBitCount
= 16; break;
233 bih
->biCompression
= amt
->subtype
.Data1
;
235 bih
->biSizeImage
= GST_VIDEO_INFO_SIZE(info
);
236 if ((vih
->AvgTimePerFrame
= (REFERENCE_TIME
)MulDiv(10000000,
237 GST_VIDEO_INFO_FPS_D(info
), GST_VIDEO_INFO_FPS_N(info
))) == -1)
238 vih
->AvgTimePerFrame
= 0; /* zero division or integer overflow */
239 bih
->biSize
= sizeof(*bih
);
240 bih
->biWidth
= width
;
241 bih
->biHeight
= height
;
246 static gboolean
amt_from_gst_caps_audio_mpeg(const GstCaps
*caps
, AM_MEDIA_TYPE
*mt
)
248 GstStructure
*structure
= gst_caps_get_structure(caps
, 0);
249 gint layer
, channels
, rate
;
251 mt
->majortype
= MEDIATYPE_Audio
;
252 mt
->subtype
= MEDIASUBTYPE_MPEG1AudioPayload
;
253 mt
->bFixedSizeSamples
= FALSE
;
254 mt
->bTemporalCompression
= FALSE
;
256 mt
->formattype
= FORMAT_WaveFormatEx
;
259 if (!gst_structure_get_int(structure
, "layer", &layer
))
261 WARN("Missing 'layer' value.\n");
264 if (!gst_structure_get_int(structure
, "channels", &channels
))
266 WARN("Missing 'channels' value.\n");
269 if (!gst_structure_get_int(structure
, "rate", &rate
))
271 WARN("Missing 'rate' value.\n");
277 MPEGLAYER3WAVEFORMAT
*wfx
= CoTaskMemAlloc(sizeof(*wfx
));
278 memset(wfx
, 0, sizeof(*wfx
));
280 mt
->subtype
.Data1
= WAVE_FORMAT_MPEGLAYER3
;
281 mt
->cbFormat
= sizeof(*wfx
);
282 mt
->pbFormat
= (BYTE
*)wfx
;
283 wfx
->wfx
.wFormatTag
= WAVE_FORMAT_MPEGLAYER3
;
284 wfx
->wfx
.nChannels
= channels
;
285 wfx
->wfx
.nSamplesPerSec
= rate
;
286 /* FIXME: We can't get most of the MPEG data from the caps. We may have
287 * to manually parse the header. */
288 wfx
->wfx
.cbSize
= sizeof(*wfx
) - sizeof(WAVEFORMATEX
);
289 wfx
->wID
= MPEGLAYER3_ID_MPEG
;
290 wfx
->fdwFlags
= MPEGLAYER3_FLAG_PADDING_ON
;
291 wfx
->nFramesPerBlock
= 1;
292 wfx
->nCodecDelay
= 1393;
296 MPEG1WAVEFORMAT
*wfx
= CoTaskMemAlloc(sizeof(*wfx
));
297 memset(wfx
, 0, sizeof(*wfx
));
299 mt
->subtype
.Data1
= WAVE_FORMAT_MPEG
;
300 mt
->cbFormat
= sizeof(*wfx
);
301 mt
->pbFormat
= (BYTE
*)wfx
;
302 wfx
->wfx
.wFormatTag
= WAVE_FORMAT_MPEG
;
303 wfx
->wfx
.nChannels
= channels
;
304 wfx
->wfx
.nSamplesPerSec
= rate
;
305 wfx
->wfx
.cbSize
= sizeof(*wfx
) - sizeof(WAVEFORMATEX
);
306 wfx
->fwHeadLayer
= layer
;
312 static gboolean
amt_from_gst_caps(const GstCaps
*caps
, AM_MEDIA_TYPE
*mt
)
314 const char *type
= gst_structure_get_name(gst_caps_get_structure(caps
, 0));
315 GstStructure
*structure
= gst_caps_get_structure(caps
, 0);
317 if (!strcmp(type
, "audio/x-raw"))
318 return amt_from_gst_caps_audio_raw(caps
, mt
);
319 else if (!strcmp(type
, "video/x-raw"))
323 if (!gst_video_info_from_caps(&info
, caps
))
325 return amt_from_gst_video_info(&info
, mt
);
327 else if (!strcmp(type
, "audio/mpeg"))
328 return amt_from_gst_caps_audio_mpeg(caps
, mt
);
329 else if (!strcmp(type
, "video/x-cinepak"))
331 VIDEOINFOHEADER
*vih
;
334 memset(mt
, 0, sizeof(AM_MEDIA_TYPE
));
335 mt
->majortype
= MEDIATYPE_Video
;
336 mt
->subtype
= MEDIASUBTYPE_CVID
;
337 mt
->bTemporalCompression
= TRUE
;
339 mt
->formattype
= FORMAT_VideoInfo
;
340 if (!(vih
= CoTaskMemAlloc(sizeof(VIDEOINFOHEADER
))))
342 mt
->cbFormat
= sizeof(VIDEOINFOHEADER
);
343 mt
->pbFormat
= (BYTE
*)vih
;
345 memset(vih
, 0, sizeof(VIDEOINFOHEADER
));
346 vih
->bmiHeader
.biSize
= sizeof(BITMAPINFOHEADER
);
347 if (gst_structure_get_int(structure
, "width", &i
))
348 vih
->bmiHeader
.biWidth
= i
;
349 if (gst_structure_get_int(structure
, "height", &i
))
350 vih
->bmiHeader
.biHeight
= i
;
351 vih
->bmiHeader
.biPlanes
= 1;
352 /* Both ffmpeg's encoder and a Cinepak file seen in the wild report
353 * 24 bpp. ffmpeg sets biSizeImage as below; others may be smaller, but
354 * as long as every sample fits into our allocator, we're fine. */
355 vih
->bmiHeader
.biBitCount
= 24;
356 vih
->bmiHeader
.biCompression
= mmioFOURCC('c','v','i','d');
357 vih
->bmiHeader
.biSizeImage
= vih
->bmiHeader
.biWidth
358 * vih
->bmiHeader
.biHeight
* vih
->bmiHeader
.biBitCount
/ 8;
363 FIXME("Unhandled type %s.\n", debugstr_a(type
));
368 static GstCaps
*amt_to_gst_caps_video(const AM_MEDIA_TYPE
*mt
)
373 GstVideoFormat format
;
377 {&MEDIASUBTYPE_ARGB32
, GST_VIDEO_FORMAT_BGRA
},
378 {&MEDIASUBTYPE_RGB32
, GST_VIDEO_FORMAT_BGRx
},
379 {&MEDIASUBTYPE_RGB24
, GST_VIDEO_FORMAT_BGR
},
380 {&MEDIASUBTYPE_RGB565
, GST_VIDEO_FORMAT_BGR16
},
381 {&MEDIASUBTYPE_RGB555
, GST_VIDEO_FORMAT_BGR15
},
384 const VIDEOINFOHEADER
*vih
= (VIDEOINFOHEADER
*)mt
->pbFormat
;
385 GstVideoFormat format
= GST_VIDEO_FORMAT_UNKNOWN
;
390 if (!IsEqualGUID(&mt
->formattype
, &FORMAT_VideoInfo
)
391 || mt
->cbFormat
< sizeof(VIDEOINFOHEADER
) || !mt
->pbFormat
)
394 for (i
= 0; i
< ARRAY_SIZE(format_map
); ++i
)
396 if (IsEqualGUID(&mt
->subtype
, format_map
[i
].subtype
))
398 format
= format_map
[i
].format
;
403 if (format
== GST_VIDEO_FORMAT_UNKNOWN
)
404 format
= gst_video_format_from_fourcc(vih
->bmiHeader
.biCompression
);
406 if (format
== GST_VIDEO_FORMAT_UNKNOWN
)
408 FIXME("Unknown video format (subtype %s, compression %#x).\n",
409 debugstr_guid(&mt
->subtype
), vih
->bmiHeader
.biCompression
);
413 gst_video_info_set_format(&info
, format
, vih
->bmiHeader
.biWidth
, vih
->bmiHeader
.biHeight
);
414 if ((caps
= gst_video_info_to_caps(&info
)))
416 /* Clear some fields that shouldn't prevent us from connecting. */
417 for (i
= 0; i
< gst_caps_get_size(caps
); ++i
)
419 gst_structure_remove_field(gst_caps_get_structure(caps
, i
), "framerate");
420 gst_structure_remove_field(gst_caps_get_structure(caps
, i
), "pixel-aspect-ratio");
426 static GstCaps
*amt_to_gst_caps_audio(const AM_MEDIA_TYPE
*mt
)
428 const WAVEFORMATEX
*wfx
= (WAVEFORMATEX
*)mt
->pbFormat
;
429 GstAudioFormat format
= GST_AUDIO_FORMAT_UNKNOWN
;
432 if (!IsEqualGUID(&mt
->formattype
, &FORMAT_WaveFormatEx
)
433 || mt
->cbFormat
< sizeof(WAVEFORMATEX
) || !mt
->pbFormat
)
436 if (IsEqualGUID(&mt
->subtype
, &MEDIASUBTYPE_PCM
))
437 format
= gst_audio_format_build_integer(wfx
->wBitsPerSample
!= 8,
438 G_LITTLE_ENDIAN
, wfx
->wBitsPerSample
, wfx
->wBitsPerSample
);
439 else if (IsEqualGUID(&mt
->subtype
, &MEDIASUBTYPE_IEEE_FLOAT
))
441 if (wfx
->wBitsPerSample
== 32)
442 format
= GST_AUDIO_FORMAT_F32LE
;
443 else if (wfx
->wBitsPerSample
== 64)
444 format
= GST_AUDIO_FORMAT_F64LE
;
447 if (format
== GST_AUDIO_FORMAT_UNKNOWN
)
449 FIXME("Unknown audio format (subtype %s, depth %u).\n",
450 debugstr_guid(&mt
->subtype
), wfx
->wBitsPerSample
);
454 gst_audio_info_set_format(&info
, format
, wfx
->nSamplesPerSec
, wfx
->nChannels
, NULL
);
455 return gst_audio_info_to_caps(&info
);
458 static GstCaps
*amt_to_gst_caps(const AM_MEDIA_TYPE
*mt
)
460 if (IsEqualGUID(&mt
->majortype
, &MEDIATYPE_Video
))
461 return amt_to_gst_caps_video(mt
);
462 else if (IsEqualGUID(&mt
->majortype
, &MEDIATYPE_Audio
))
463 return amt_to_gst_caps_audio(mt
);
465 FIXME("Unknown major type %s.\n", debugstr_guid(&mt
->majortype
));
469 static gboolean
setcaps_sink(GstPad
*pad
, GstCaps
*caps
)
471 struct gstdemux_source
*pin
= gst_pad_get_element_private(pad
);
472 struct gstdemux
*filter
= impl_from_strmbase_filter(pin
->pin
.pin
.filter
);
473 gchar
*caps_str
= gst_caps_to_string(caps
);
475 TRACE("filter %p, caps %s.\n", filter
, debugstr_a(caps_str
));
478 FreeMediaType(&pin
->mt
);
480 if (!amt_from_gst_caps(caps
, &pin
->mt
))
483 SetEvent(pin
->caps_event
);
487 static gboolean
query_sink(GstPad
*pad
, GstObject
*parent
, GstQuery
*query
)
489 struct gstdemux_source
*pin
= gst_pad_get_element_private(pad
);
491 TRACE("pin %p, type \"%s\".\n", pin
, gst_query_type_get_name(query
->type
));
497 GstCaps
*caps
, *filter
, *temp
;
499 gst_query_parse_caps(query
, &filter
);
501 if (pin
->pin
.pin
.peer
)
502 caps
= amt_to_gst_caps(&pin
->pin
.pin
.mt
);
504 caps
= gst_caps_new_any();
510 temp
= gst_caps_intersect(caps
, filter
);
511 gst_caps_unref(caps
);
515 gst_query_set_caps_result(query
, caps
);
516 gst_caps_unref(caps
);
519 case GST_QUERY_ACCEPT_CAPS
:
525 if (!pin
->pin
.pin
.peer
)
527 gst_query_set_accept_caps_result(query
, TRUE
);
531 gst_query_parse_accept_caps(query
, &caps
);
532 if (!amt_from_gst_caps(caps
, &mt
))
535 if (!IsEqualGUID(&mt
.majortype
, &pin
->pin
.pin
.mt
.majortype
)
536 || !IsEqualGUID(&mt
.subtype
, &pin
->pin
.pin
.mt
.subtype
)
537 || !IsEqualGUID(&mt
.formattype
, &pin
->pin
.pin
.mt
.formattype
))
540 if (IsEqualGUID(&mt
.majortype
, &MEDIATYPE_Video
))
542 const VIDEOINFOHEADER
*req_vih
= (VIDEOINFOHEADER
*)mt
.pbFormat
;
543 const VIDEOINFOHEADER
*our_vih
= (VIDEOINFOHEADER
*)pin
->pin
.pin
.mt
.pbFormat
;
545 if (req_vih
->bmiHeader
.biWidth
!= our_vih
->bmiHeader
.biWidth
546 || req_vih
->bmiHeader
.biHeight
!= our_vih
->bmiHeader
.biHeight
547 || req_vih
->bmiHeader
.biBitCount
!= our_vih
->bmiHeader
.biBitCount
548 || req_vih
->bmiHeader
.biCompression
!= our_vih
->bmiHeader
.biCompression
)
551 else if (IsEqualGUID(&mt
.majortype
, &MEDIATYPE_Audio
))
553 const WAVEFORMATEX
*req_wfx
= (WAVEFORMATEX
*)mt
.pbFormat
;
554 const WAVEFORMATEX
*our_wfx
= (WAVEFORMATEX
*)pin
->pin
.pin
.mt
.pbFormat
;
556 if (req_wfx
->nChannels
!= our_wfx
->nChannels
557 || req_wfx
->nSamplesPerSec
!= our_wfx
->nSamplesPerSec
558 || req_wfx
->wBitsPerSample
!= our_wfx
->wBitsPerSample
)
564 if (!ret
&& WARN_ON(gstreamer
))
566 gchar
*str
= gst_caps_to_string(caps
);
567 WARN("Rejecting caps \"%s\".\n", debugstr_a(str
));
571 gst_query_set_accept_caps_result(query
, ret
);
575 return gst_pad_query_default (pad
, parent
, query
);
579 static gboolean
gst_base_src_perform_seek(struct gstdemux
*This
, GstEvent
*event
)
583 GstFormat seek_format
;
585 GstSeekType cur_type
, stop_type
;
590 BOOL thread
= !!This
->push_thread
;
592 gst_event_parse_seek(event
, &rate
, &seek_format
, &flags
,
593 &cur_type
, &cur
, &stop_type
, &stop
);
595 if (seek_format
!= GST_FORMAT_BYTES
)
597 FIXME("Unhandled format \"%s\".\n", gst_format_get_name(seek_format
));
601 flush
= flags
& GST_SEEK_FLAG_FLUSH
;
602 seqnum
= gst_event_get_seqnum(event
);
604 /* send flush start */
606 tevent
= gst_event_new_flush_start();
607 gst_event_set_seqnum(tevent
, seqnum
);
608 gst_pad_push_event(This
->my_src
, tevent
);
610 IAsyncReader_BeginFlush(This
->reader
);
612 gst_pad_set_active(This
->my_src
, 1);
615 This
->nextofs
= This
->start
= cur
;
617 /* and prepare to continue streaming */
619 tevent
= gst_event_new_flush_stop(TRUE
);
620 gst_event_set_seqnum(tevent
, seqnum
);
621 gst_pad_push_event(This
->my_src
, tevent
);
623 IAsyncReader_EndFlush(This
->reader
);
625 gst_pad_set_active(This
->my_src
, 1);
631 static gboolean
event_src(GstPad
*pad
, GstObject
*parent
, GstEvent
*event
)
633 struct gstdemux
*This
= gst_pad_get_element_private(pad
);
635 TRACE("filter %p, type \"%s\".\n", This
, GST_EVENT_TYPE_NAME(event
));
637 switch (event
->type
) {
639 return gst_base_src_perform_seek(This
, event
);
640 case GST_EVENT_FLUSH_START
:
641 EnterCriticalSection(&This
->filter
.csFilter
);
643 IAsyncReader_BeginFlush(This
->reader
);
644 LeaveCriticalSection(&This
->filter
.csFilter
);
646 case GST_EVENT_FLUSH_STOP
:
647 EnterCriticalSection(&This
->filter
.csFilter
);
649 IAsyncReader_EndFlush(This
->reader
);
650 LeaveCriticalSection(&This
->filter
.csFilter
);
653 WARN("Ignoring \"%s\" event.\n", GST_EVENT_TYPE_NAME(event
));
656 case GST_EVENT_RECONFIGURE
:
657 return gst_pad_event_default(pad
, parent
, event
);
662 static gboolean
event_sink(GstPad
*pad
, GstObject
*parent
, GstEvent
*event
)
664 struct gstdemux_source
*pin
= gst_pad_get_element_private(pad
);
666 TRACE("pin %p, type \"%s\".\n", pin
, GST_EVENT_TYPE_NAME(event
));
668 switch (event
->type
) {
669 case GST_EVENT_SEGMENT
: {
670 gdouble rate
, applied_rate
;
672 const GstSegment
*segment
;
674 gst_event_parse_segment(event
, &segment
);
676 pos
= segment
->position
;
677 stop
= segment
->stop
;
678 rate
= segment
->rate
;
679 applied_rate
= segment
->applied_rate
;
681 if (segment
->format
!= GST_FORMAT_TIME
)
683 FIXME("Unhandled format \"%s\".\n", gst_format_get_name(segment
->format
));
687 gst_segment_copy_into(segment
, pin
->segment
);
694 if (pin
->pin
.pin
.peer
)
695 IPin_NewSegment(pin
->pin
.pin
.peer
, pos
, stop
, rate
*applied_rate
);
700 if (pin
->pin
.pin
.peer
)
701 IPin_EndOfStream(pin
->pin
.pin
.peer
);
703 SetEvent(pin
->eos_event
);
705 case GST_EVENT_FLUSH_START
:
706 if (impl_from_strmbase_filter(pin
->pin
.pin
.filter
)->ignore_flush
) {
707 /* gst-plugins-base prior to 1.7 contains a bug which causes
708 * our sink pins to receive a flush-start event when the
709 * decodebin changes from PAUSED to READY (including
710 * PLAYING->PAUSED->READY), but no matching flush-stop event is
711 * sent. See <gst-plugins-base.git:60bad4815db966a8e4). Here we
712 * unset the flushing flag to avoid the problem. */
713 TRACE("Working around gst <1.7 bug, ignoring FLUSH_START\n");
714 GST_PAD_UNSET_FLUSHING (pad
);
717 if (pin
->pin
.pin
.peer
)
718 IPin_BeginFlush(pin
->pin
.pin
.peer
);
720 case GST_EVENT_FLUSH_STOP
:
721 gst_segment_init(pin
->segment
, GST_FORMAT_TIME
);
722 if (pin
->pin
.pin
.peer
)
723 IPin_EndFlush(pin
->pin
.pin
.peer
);
725 case GST_EVENT_CAPS
: {
727 gst_event_parse_caps(event
, &caps
);
728 return setcaps_sink(pad
, caps
);
731 WARN("Ignoring \"%s\" event.\n", GST_EVENT_TYPE_NAME(event
));
732 return gst_pad_event_default(pad
, parent
, event
);
736 static DWORD CALLBACK
push_data(LPVOID iface
)
738 LONGLONG maxlen
, curlen
;
739 struct gstdemux
*This
= iface
;
744 if (!(buffer
= gst_buffer_new_allocate(NULL
, 16384, NULL
)))
746 ERR("Failed to allocate memory.\n");
750 IBaseFilter_AddRef(&This
->filter
.IBaseFilter_iface
);
753 IAsyncReader_Length(This
->reader
, &maxlen
, &curlen
);
757 TRACE("Starting..\n");
762 if (This
->nextofs
>= maxlen
)
764 len
= min(16384, maxlen
- This
->nextofs
);
766 if (!gst_buffer_map_range(buffer
, -1, len
, &mapping
, GST_MAP_WRITE
))
768 ERR("Failed to map buffer.\n");
771 hr
= IAsyncReader_SyncRead(This
->reader
, This
->nextofs
, len
, mapping
.data
);
772 gst_buffer_unmap(buffer
, &mapping
);
775 ERR("Failed to read data, hr %#x.\n", hr
);
779 This
->nextofs
+= len
;
781 buffer
->duration
= buffer
->pts
= -1;
782 ret
= gst_pad_push(This
->my_src
, buffer
);
786 ERR("Sending returned: %i\n", ret
);
787 if (ret
== GST_FLOW_ERROR
)
789 else if (ret
== GST_FLOW_FLUSHING
)
790 hr
= VFW_E_WRONG_STATE
;
795 gst_buffer_unref(buffer
);
797 gst_pad_push_event(This
->my_src
, gst_event_new_eos());
799 TRACE("Stopping.. %08x\n", hr
);
801 IBaseFilter_Release(&This
->filter
.IBaseFilter_iface
);
806 static GstFlowReturn
got_data_sink(GstPad
*pad
, GstObject
*parent
, GstBuffer
*buf
)
808 struct gstdemux_source
*pin
= gst_pad_get_element_private(pad
);
809 struct gstdemux
*This
= impl_from_strmbase_filter(pin
->pin
.pin
.filter
);
812 IMediaSample
*sample
;
815 TRACE("%p %p\n", pad
, buf
);
818 gst_buffer_unref(buf
);
822 hr
= BaseOutputPinImpl_GetDeliveryBuffer(&pin
->pin
, &sample
, NULL
, NULL
, 0);
824 if (hr
== VFW_E_NOT_CONNECTED
) {
825 gst_buffer_unref(buf
);
826 return GST_FLOW_NOT_LINKED
;
830 gst_buffer_unref(buf
);
831 ERR("Could not get a delivery buffer (%x), returning GST_FLOW_FLUSHING\n", hr
);
832 return GST_FLOW_FLUSHING
;
835 gst_buffer_map(buf
, &info
, GST_MAP_READ
);
837 hr
= IMediaSample_SetActualDataLength(sample
, info
.size
);
839 WARN("SetActualDataLength failed: %08x\n", hr
);
840 return GST_FLOW_FLUSHING
;
843 IMediaSample_GetPointer(sample
, &ptr
);
845 memcpy(ptr
, info
.data
, info
.size
);
847 gst_buffer_unmap(buf
, &info
);
849 if (GST_BUFFER_PTS_IS_VALID(buf
)) {
850 REFERENCE_TIME rtStart
= gst_segment_to_running_time(pin
->segment
, GST_FORMAT_TIME
, buf
->pts
);
854 if (GST_BUFFER_DURATION_IS_VALID(buf
)) {
855 REFERENCE_TIME tStart
= buf
->pts
/ 100;
856 REFERENCE_TIME tStop
= (buf
->pts
+ buf
->duration
) / 100;
857 REFERENCE_TIME rtStop
;
858 rtStop
= gst_segment_to_running_time(pin
->segment
, GST_FORMAT_TIME
, buf
->pts
+ buf
->duration
);
861 TRACE("Current time on %p: %i to %i ms\n", pin
, (int)(rtStart
/ 10000), (int)(rtStop
/ 10000));
862 IMediaSample_SetTime(sample
, &rtStart
, rtStop
>= 0 ? &rtStop
: NULL
);
863 IMediaSample_SetMediaTime(sample
, &tStart
, &tStop
);
865 IMediaSample_SetTime(sample
, rtStart
>= 0 ? &rtStart
: NULL
, NULL
);
866 IMediaSample_SetMediaTime(sample
, NULL
, NULL
);
869 IMediaSample_SetTime(sample
, NULL
, NULL
);
870 IMediaSample_SetMediaTime(sample
, NULL
, NULL
);
873 IMediaSample_SetDiscontinuity(sample
, GST_BUFFER_FLAG_IS_SET(buf
, GST_BUFFER_FLAG_DISCONT
));
874 IMediaSample_SetPreroll(sample
, GST_BUFFER_FLAG_IS_SET(buf
, GST_BUFFER_FLAG_LIVE
));
875 IMediaSample_SetSyncPoint(sample
, !GST_BUFFER_FLAG_IS_SET(buf
, GST_BUFFER_FLAG_DELTA_UNIT
));
877 if (!pin
->pin
.pin
.peer
)
878 hr
= VFW_E_NOT_CONNECTED
;
880 hr
= IMemInputPin_Receive(pin
->pin
.pMemInputPin
, sample
);
882 TRACE("sending sample returned: %08x\n", hr
);
884 gst_buffer_unref(buf
);
885 IMediaSample_Release(sample
);
887 if (hr
== VFW_E_NOT_CONNECTED
)
888 return GST_FLOW_NOT_LINKED
;
891 return GST_FLOW_FLUSHING
;
896 static GstFlowReturn
request_buffer_src(GstPad
*pad
, GstObject
*parent
, guint64 ofs
, guint len
, GstBuffer
**buf
)
898 struct gstdemux
*This
= gst_pad_get_element_private(pad
);
902 TRACE("pad %p, offset %s, length %u, buffer %p.\n", pad
, wine_dbgstr_longlong(ofs
), len
, *buf
);
904 if (ofs
== GST_BUFFER_OFFSET_NONE
)
905 ofs
= This
->nextpullofs
;
906 if (ofs
>= This
->filesize
) {
907 WARN("Reading past eof: %s, %u\n", wine_dbgstr_longlong(ofs
), len
);
910 if (len
+ ofs
> This
->filesize
)
911 len
= This
->filesize
- ofs
;
912 This
->nextpullofs
= ofs
+ len
;
915 *buf
= gst_buffer_new_and_alloc(len
);
916 gst_buffer_map(*buf
, &info
, GST_MAP_WRITE
);
917 hr
= IAsyncReader_SyncRead(This
->reader
, ofs
, len
, info
.data
);
918 gst_buffer_unmap(*buf
, &info
);
920 ERR("Returned %08x\n", hr
);
921 return GST_FLOW_ERROR
;
924 GST_BUFFER_OFFSET(*buf
) = ofs
;
928 static DWORD CALLBACK
push_data_init(LPVOID iface
)
930 struct gstdemux
*This
= iface
;
933 TRACE("Starting..\n");
936 GstFlowReturn ret
= request_buffer_src(This
->my_src
, NULL
, ofs
, 4096, &buf
);
938 ERR("Obtaining buffer returned: %i\n", ret
);
941 ret
= gst_pad_push(This
->my_src
, buf
);
944 TRACE("Sending returned: %i\n", ret
);
948 TRACE("Stopping..\n");
952 static void removed_decoded_pad(GstElement
*bin
, GstPad
*pad
, gpointer user
)
954 struct gstdemux
*filter
= user
;
958 TRACE("filter %p, bin %p, pad %p.\n", filter
, bin
, pad
);
960 for (i
= 0; i
< filter
->source_count
; ++i
)
962 struct gstdemux_source
*pin
= filter
->sources
[i
];
964 if (pin
->their_src
== pad
)
967 gst_pad_unlink(pin
->their_src
, pin
->post_sink
);
969 gst_pad_unlink(pin
->their_src
, pin
->my_sink
);
970 gst_object_unref(pin
->their_src
);
971 pin
->their_src
= NULL
;
976 name
= gst_pad_get_name(pad
);
977 WARN("No pin matching pad %s found.\n", debugstr_a(name
));
981 static void init_new_decoded_pad(GstElement
*bin
, GstPad
*pad
, struct gstdemux
*This
)
983 static const WCHAR formatW
[] = {'S','t','r','e','a','m',' ','%','0','2','u',0};
984 const char *typename
;
988 struct gstdemux_source
*pin
;
992 TRACE("%p %p %p\n", This
, bin
, pad
);
994 sprintfW(nameW
, formatW
, This
->source_count
);
996 name
= gst_pad_get_name(pad
);
997 TRACE("Name: %s\n", name
);
1000 caps
= gst_pad_query_caps(pad
, NULL
);
1001 caps
= gst_caps_make_writable(caps
);
1002 arg
= gst_caps_get_structure(caps
, 0);
1003 typename
= gst_structure_get_name(arg
);
1005 if (!(pin
= create_pin(This
, nameW
)))
1007 ERR("Failed to allocate memory.\n");
1011 if (!strcmp(typename
, "video/x-raw"))
1013 GstElement
*vconv
, *flip
;
1015 /* decodebin considers many YUV formats to be "raw", but some quartz
1016 * filters can't handle those. Also, videoflip can't handle all "raw"
1017 * formats either. Add a videoconvert to swap color spaces. */
1018 if (!(vconv
= gst_element_factory_make("videoconvert", NULL
)))
1020 ERR("Failed to create videoconvert, are %u-bit GStreamer \"base\" plugins installed?\n",
1021 8 * (int)sizeof(void *));
1025 /* GStreamer outputs RGB video top-down, but DirectShow expects bottom-up. */
1026 if (!(flip
= gst_element_factory_make("videoflip", NULL
)))
1028 ERR("Failed to create videoflip, are %u-bit GStreamer \"good\" plugins installed?\n",
1029 8 * (int)sizeof(void *));
1033 gst_bin_add(GST_BIN(This
->container
), vconv
); /* bin takes ownership */
1034 gst_element_sync_state_with_parent(vconv
);
1035 gst_bin_add(GST_BIN(This
->container
), flip
); /* bin takes ownership */
1036 gst_element_sync_state_with_parent(flip
);
1038 gst_element_link(vconv
, flip
);
1040 pin
->post_sink
= gst_element_get_static_pad(vconv
, "sink");
1041 pin
->post_src
= gst_element_get_static_pad(flip
, "src");
1044 else if (!strcmp(typename
, "audio/x-raw"))
1046 GstElement
*convert
;
1048 /* Currently our dsound can't handle 64-bit formats or all
1049 * surround-sound configurations. Native dsound can't always handle
1050 * 64-bit formats either. Add an audioconvert to allow changing bit
1051 * depth and channel count. */
1052 if (!(convert
= gst_element_factory_make("audioconvert", NULL
)))
1054 ERR("Failed to create audioconvert, are %u-bit GStreamer \"base\" plugins installed?\n",
1055 8 * (int)sizeof(void *));
1059 gst_bin_add(GST_BIN(This
->container
), convert
);
1060 gst_element_sync_state_with_parent(convert
);
1062 pin
->post_sink
= gst_element_get_static_pad(convert
, "sink");
1063 pin
->post_src
= gst_element_get_static_pad(convert
, "src");
1068 if ((ret
= gst_pad_link(pad
, pin
->post_sink
)) < 0)
1070 ERR("Failed to link decodebin source pad to post-processing elements, error %s.\n",
1071 gst_pad_link_get_name(ret
));
1072 gst_object_unref(pin
->post_sink
);
1073 pin
->post_sink
= NULL
;
1077 if ((ret
= gst_pad_link(pin
->post_src
, pin
->my_sink
)) < 0)
1079 ERR("Failed to link post-processing elements to our sink pad, error %s.\n",
1080 gst_pad_link_get_name(ret
));
1081 gst_object_unref(pin
->post_src
);
1082 pin
->post_src
= NULL
;
1083 gst_object_unref(pin
->post_sink
);
1084 pin
->post_sink
= NULL
;
1088 else if ((ret
= gst_pad_link(pad
, pin
->my_sink
)) < 0)
1090 ERR("Failed to link decodebin source pad to our sink pad, error %s.\n",
1091 gst_pad_link_get_name(ret
));
1095 gst_pad_set_active(pin
->my_sink
, 1);
1096 gst_object_ref(pin
->their_src
= pad
);
1098 gst_caps_unref(caps
);
1101 static void existing_new_pad(GstElement
*bin
, GstPad
*pad
, gpointer user
)
1103 struct gstdemux
*This
= user
;
1107 TRACE("%p %p %p\n", This
, bin
, pad
);
1109 if (gst_pad_is_linked(pad
))
1112 /* Still holding our own lock */
1113 if (This
->initial
) {
1114 init_new_decoded_pad(bin
, pad
, This
);
1118 for (i
= 0; i
< This
->source_count
; ++i
)
1120 struct gstdemux_source
*pin
= This
->sources
[i
];
1121 if (!pin
->their_src
) {
1122 gst_segment_init(pin
->segment
, GST_FORMAT_TIME
);
1125 ret
= gst_pad_link(pad
, pin
->post_sink
);
1127 ret
= gst_pad_link(pad
, pin
->my_sink
);
1130 pin
->their_src
= pad
;
1131 gst_object_ref(pin
->their_src
);
1132 TRACE("Relinked\n");
1137 init_new_decoded_pad(bin
, pad
, This
);
1140 static gboolean
query_function(GstPad
*pad
, GstObject
*parent
, GstQuery
*query
)
1142 struct gstdemux
*This
= gst_pad_get_element_private(pad
);
1145 TRACE("filter %p, type %s.\n", This
, GST_QUERY_TYPE_NAME(query
));
1147 switch (GST_QUERY_TYPE(query
)) {
1148 case GST_QUERY_DURATION
:
1149 gst_query_parse_duration(query
, &format
, NULL
);
1150 if (format
== GST_FORMAT_PERCENT
)
1152 gst_query_set_duration(query
, GST_FORMAT_PERCENT
, GST_FORMAT_PERCENT_MAX
);
1155 else if (format
== GST_FORMAT_BYTES
)
1157 gst_query_set_duration(query
, GST_FORMAT_BYTES
, This
->filesize
);
1161 case GST_QUERY_SEEKING
:
1162 gst_query_parse_seeking (query
, &format
, NULL
, NULL
, NULL
);
1163 if (format
!= GST_FORMAT_BYTES
)
1165 WARN("Cannot seek using format \"%s\".\n", gst_format_get_name(format
));
1168 gst_query_set_seeking(query
, GST_FORMAT_BYTES
, 1, 0, This
->filesize
);
1170 case GST_QUERY_SCHEDULING
:
1171 gst_query_set_scheduling(query
, GST_SCHEDULING_FLAG_SEEKABLE
, 1, -1, 0);
1172 gst_query_add_scheduling_mode(query
, GST_PAD_MODE_PUSH
);
1173 gst_query_add_scheduling_mode(query
, GST_PAD_MODE_PULL
);
1176 WARN("Unhandled query type %s.\n", GST_QUERY_TYPE_NAME(query
));
1181 static gboolean
activate_push(GstPad
*pad
, gboolean activate
)
1183 struct gstdemux
*This
= gst_pad_get_element_private(pad
);
1185 EnterCriticalSection(&This
->filter
.csFilter
);
1187 TRACE("Deactivating\n");
1189 IAsyncReader_BeginFlush(This
->reader
);
1190 if (This
->push_thread
) {
1191 WaitForSingleObject(This
->push_thread
, -1);
1192 CloseHandle(This
->push_thread
);
1193 This
->push_thread
= NULL
;
1196 IAsyncReader_EndFlush(This
->reader
);
1197 if (This
->filter
.state
== State_Stopped
)
1198 This
->nextofs
= This
->start
;
1199 } else if (!This
->push_thread
) {
1200 TRACE("Activating\n");
1202 This
->push_thread
= CreateThread(NULL
, 0, push_data_init
, This
, 0, NULL
);
1204 This
->push_thread
= CreateThread(NULL
, 0, push_data
, This
, 0, NULL
);
1206 LeaveCriticalSection(&This
->filter
.csFilter
);
1210 static gboolean
activate_mode(GstPad
*pad
, GstObject
*parent
, GstPadMode mode
, gboolean activate
)
1212 struct gstdemux
*filter
= gst_pad_get_element_private(pad
);
1214 TRACE("%s source pad for filter %p in %s mode.\n",
1215 activate
? "Activating" : "Deactivating", filter
, gst_pad_mode_get_name(mode
));
1218 case GST_PAD_MODE_PULL
:
1220 case GST_PAD_MODE_PUSH
:
1221 return activate_push(pad
, activate
);
1228 static void no_more_pads(GstElement
*decodebin
, gpointer user
)
1230 struct gstdemux
*filter
= user
;
1231 TRACE("filter %p.\n", filter
);
1232 SetEvent(filter
->no_more_pads_event
);
1235 static GstAutoplugSelectResult
autoplug_blacklist(GstElement
*bin
, GstPad
*pad
, GstCaps
*caps
, GstElementFactory
*fact
, gpointer user
)
1237 const char *name
= gst_element_factory_get_longname(fact
);
1239 if (strstr(name
, "Player protection")) {
1240 WARN("Blacklisted a/52 decoder because it only works in Totem\n");
1241 return GST_AUTOPLUG_SELECT_SKIP
;
1243 if (!strcmp(name
, "Fluendo Hardware Accelerated Video Decoder")) {
1244 WARN("Disabled video acceleration since it breaks in wine\n");
1245 return GST_AUTOPLUG_SELECT_SKIP
;
1247 TRACE("using \"%s\"\n", name
);
1248 return GST_AUTOPLUG_SELECT_TRY
;
1251 static GstBusSyncReply
watch_bus(GstBus
*bus
, GstMessage
*msg
, gpointer data
)
1253 struct gstdemux
*filter
= data
;
1255 gchar
*dbg_info
= NULL
;
1257 TRACE("filter %p, message type %s.\n", filter
, GST_MESSAGE_TYPE_NAME(msg
));
1261 case GST_MESSAGE_ERROR
:
1262 gst_message_parse_error(msg
, &err
, &dbg_info
);
1263 ERR("%s: %s\n", GST_OBJECT_NAME(msg
->src
), err
->message
);
1264 ERR("%s\n", dbg_info
);
1267 SetEvent(filter
->error_event
);
1269 case GST_MESSAGE_WARNING
:
1270 gst_message_parse_warning(msg
, &err
, &dbg_info
);
1271 WARN("%s: %s\n", GST_OBJECT_NAME(msg
->src
), err
->message
);
1272 WARN("%s\n", dbg_info
);
1276 case GST_MESSAGE_DURATION_CHANGED
:
1277 SetEvent(filter
->duration_event
);
1282 return GST_BUS_DROP
;
1285 static void unknown_type(GstElement
*bin
, GstPad
*pad
, GstCaps
*caps
, gpointer user
)
1287 gchar
*strcaps
= gst_caps_to_string(caps
);
1288 ERR("Could not find a filter for caps: %s\n", debugstr_a(strcaps
));
1292 static HRESULT
GST_Connect(struct gstdemux
*This
, IPin
*pConnectPin
)
1295 GstStaticPadTemplate src_template
= GST_STATIC_PAD_TEMPLATE(
1299 GST_STATIC_CAPS_ANY
);
1301 IAsyncReader_Length(This
->reader
, &This
->filesize
, &avail
);
1304 This
->bus
= gst_bus_new();
1305 gst_bus_set_sync_handler(This
->bus
, watch_bus_wrapper
, This
, NULL
);
1308 This
->container
= gst_bin_new(NULL
);
1309 gst_element_set_bus(This
->container
, This
->bus
);
1311 This
->my_src
= gst_pad_new_from_static_template(&src_template
, "quartz-src");
1312 gst_pad_set_getrange_function(This
->my_src
, request_buffer_src_wrapper
);
1313 gst_pad_set_query_function(This
->my_src
, query_function_wrapper
);
1314 gst_pad_set_activatemode_function(This
->my_src
, activate_mode_wrapper
);
1315 gst_pad_set_event_function(This
->my_src
, event_src_wrapper
);
1316 gst_pad_set_element_private (This
->my_src
, This
);
1318 This
->start
= This
->nextofs
= This
->nextpullofs
= This
->stop
= 0;
1320 This
->initial
= TRUE
;
1321 if (!This
->init_gst(This
))
1323 This
->initial
= FALSE
;
1325 This
->nextofs
= This
->nextpullofs
= 0;
1329 static LONGLONG
query_duration(GstPad
*pad
)
1331 gint64 duration
, byte_length
;
1333 if (gst_pad_query_duration(pad
, GST_FORMAT_TIME
, &duration
))
1334 return duration
/ 100;
1336 WARN("Failed to query time duration; trying to convert from byte length.\n");
1338 /* To accurately get a duration for the stream, we want to only consider the
1339 * length of that stream. Hence, query for the pad duration, instead of
1340 * using the file duration. */
1341 if (gst_pad_query_duration(pad
, GST_FORMAT_BYTES
, &byte_length
)
1342 && gst_pad_query_convert(pad
, GST_FORMAT_BYTES
, byte_length
, GST_FORMAT_TIME
, &duration
))
1343 return duration
/ 100;
1345 ERR("Failed to query duration.\n");
1349 static inline struct gstdemux_source
*impl_from_IMediaSeeking(IMediaSeeking
*iface
)
1351 return CONTAINING_RECORD(iface
, struct gstdemux_source
, seek
.IMediaSeeking_iface
);
1354 static struct strmbase_pin
*gstdemux_get_pin(struct strmbase_filter
*base
, unsigned int index
)
1356 struct gstdemux
*filter
= impl_from_strmbase_filter(base
);
1358 if (filter
->enum_sink_first
)
1361 return &filter
->sink
.pin
;
1362 else if (index
<= filter
->source_count
)
1363 return &filter
->sources
[index
- 1]->pin
.pin
;
1367 if (index
< filter
->source_count
)
1368 return &filter
->sources
[index
]->pin
.pin
;
1369 else if (index
== filter
->source_count
)
1370 return &filter
->sink
.pin
;
1375 static void gstdemux_destroy(struct strmbase_filter
*iface
)
1377 struct gstdemux
*filter
= impl_from_strmbase_filter(iface
);
1380 CloseHandle(filter
->no_more_pads_event
);
1381 CloseHandle(filter
->duration_event
);
1382 CloseHandle(filter
->error_event
);
1384 /* Don't need to clean up output pins, disconnecting input pin will do that */
1385 if (filter
->sink
.pin
.peer
)
1387 hr
= IPin_Disconnect(filter
->sink
.pin
.peer
);
1389 hr
= IPin_Disconnect(&filter
->sink
.pin
.IPin_iface
);
1394 IAsyncReader_Release(filter
->reader
);
1395 filter
->reader
= NULL
;
1399 gst_bus_set_sync_handler(filter
->bus
, NULL
, NULL
, NULL
);
1400 gst_object_unref(filter
->bus
);
1402 strmbase_sink_cleanup(&filter
->sink
);
1403 strmbase_filter_cleanup(&filter
->filter
);
1407 static HRESULT
gstdemux_init_stream(struct strmbase_filter
*iface
)
1409 struct gstdemux
*filter
= impl_from_strmbase_filter(iface
);
1410 HRESULT hr
= VFW_E_NOT_CONNECTED
, pin_hr
;
1411 const SourceSeeking
*seeking
;
1412 GstStateChangeReturn ret
;
1415 if (!filter
->container
)
1416 return VFW_E_NOT_CONNECTED
;
1418 for (i
= 0; i
< filter
->source_count
; ++i
)
1420 if (SUCCEEDED(pin_hr
= BaseOutputPinImpl_Active(&filter
->sources
[i
]->pin
)))
1427 if (filter
->no_more_pads_event
)
1428 ResetEvent(filter
->no_more_pads_event
);
1430 if ((ret
= gst_element_set_state(filter
->container
, GST_STATE_PAUSED
)) == GST_STATE_CHANGE_FAILURE
)
1432 ERR("Failed to pause stream.\n");
1436 /* Make sure that all of our pads are connected before returning, lest we
1437 * e.g. try to seek and fail. */
1438 if (filter
->no_more_pads_event
)
1439 WaitForSingleObject(filter
->no_more_pads_event
, INFINITE
);
1441 seeking
= &filter
->sources
[0]->seek
;
1443 /* GStreamer can't seek while stopped, and it resets position to the
1444 * beginning of the stream every time it is stopped. */
1445 if (seeking
->llCurrent
)
1447 GstSeekType stop_type
= GST_SEEK_TYPE_NONE
;
1449 if (seeking
->llStop
&& seeking
->llStop
!= seeking
->llDuration
)
1450 stop_type
= GST_SEEK_TYPE_SET
;
1452 gst_pad_push_event(filter
->sources
[0]->my_sink
, gst_event_new_seek(
1453 seeking
->dRate
, GST_FORMAT_TIME
, GST_SEEK_FLAG_FLUSH
,
1454 GST_SEEK_TYPE_SET
, seeking
->llCurrent
* 100,
1455 stop_type
, seeking
->llStop
* 100));
1461 static HRESULT
gstdemux_start_stream(struct strmbase_filter
*iface
, REFERENCE_TIME time
)
1463 struct gstdemux
*filter
= impl_from_strmbase_filter(iface
);
1464 GstStateChangeReturn ret
;
1466 if (!filter
->container
)
1467 return VFW_E_NOT_CONNECTED
;
1469 if ((ret
= gst_element_set_state(filter
->container
, GST_STATE_PLAYING
)) == GST_STATE_CHANGE_FAILURE
)
1471 ERR("Failed to play stream.\n");
1474 else if (ret
== GST_STATE_CHANGE_ASYNC
)
1479 static HRESULT
gstdemux_stop_stream(struct strmbase_filter
*iface
)
1481 struct gstdemux
*filter
= impl_from_strmbase_filter(iface
);
1482 GstStateChangeReturn ret
;
1484 if (!filter
->container
)
1485 return VFW_E_NOT_CONNECTED
;
1487 if ((ret
= gst_element_set_state(filter
->container
, GST_STATE_PAUSED
)) == GST_STATE_CHANGE_FAILURE
)
1489 ERR("Failed to pause stream.\n");
1492 else if (ret
== GST_STATE_CHANGE_ASYNC
)
1497 static HRESULT
gstdemux_cleanup_stream(struct strmbase_filter
*iface
)
1499 struct gstdemux
*filter
= impl_from_strmbase_filter(iface
);
1500 GstStateChangeReturn ret
;
1503 if (!filter
->container
)
1506 filter
->ignore_flush
= TRUE
;
1507 if ((ret
= gst_element_set_state(filter
->container
, GST_STATE_READY
)) == GST_STATE_CHANGE_FAILURE
)
1509 ERR("Failed to pause stream.\n");
1512 gst_element_get_state(filter
->container
, NULL
, NULL
, GST_CLOCK_TIME_NONE
);
1513 filter
->ignore_flush
= FALSE
;
1515 for (i
= 0; i
< filter
->source_count
; ++i
)
1517 if (filter
->sources
[i
]->pin
.pin
.peer
)
1518 IMemAllocator_Decommit(filter
->sources
[i
]->pin
.pAllocator
);
1524 static HRESULT
gstdemux_wait_state(struct strmbase_filter
*iface
, DWORD timeout
)
1526 struct gstdemux
*filter
= impl_from_strmbase_filter(iface
);
1527 GstStateChangeReturn ret
;
1529 if (!filter
->container
)
1532 ret
= gst_element_get_state(filter
->container
, NULL
, NULL
,
1533 timeout
== INFINITE
? GST_CLOCK_TIME_NONE
: timeout
* 1000000);
1534 if (ret
== GST_STATE_CHANGE_FAILURE
)
1536 ERR("Failed to get state.\n");
1539 else if (ret
== GST_STATE_CHANGE_ASYNC
)
1540 return VFW_S_STATE_INTERMEDIATE
;
1544 static const struct strmbase_filter_ops filter_ops
=
1546 .filter_get_pin
= gstdemux_get_pin
,
1547 .filter_destroy
= gstdemux_destroy
,
1548 .filter_init_stream
= gstdemux_init_stream
,
1549 .filter_start_stream
= gstdemux_start_stream
,
1550 .filter_stop_stream
= gstdemux_stop_stream
,
1551 .filter_cleanup_stream
= gstdemux_cleanup_stream
,
1552 .filter_wait_state
= gstdemux_wait_state
,
1555 static inline struct gstdemux
*impl_from_strmbase_sink(struct strmbase_sink
*iface
)
1557 return CONTAINING_RECORD(iface
, struct gstdemux
, sink
);
1560 static HRESULT
sink_query_accept(struct strmbase_pin
*iface
, const AM_MEDIA_TYPE
*mt
)
1562 if (IsEqualGUID(&mt
->majortype
, &MEDIATYPE_Stream
))
1567 static HRESULT
gstdemux_sink_connect(struct strmbase_sink
*iface
, IPin
*peer
, const AM_MEDIA_TYPE
*pmt
)
1569 struct gstdemux
*filter
= impl_from_strmbase_sink(iface
);
1574 filter
->reader
= NULL
;
1575 if (FAILED(hr
= IPin_QueryInterface(peer
, &IID_IAsyncReader
, (void **)&filter
->reader
)))
1578 if (FAILED(hr
= GST_Connect(filter
, peer
)))
1583 GST_RemoveOutputPins(filter
);
1584 IAsyncReader_Release(filter
->reader
);
1585 filter
->reader
= NULL
;
1589 static void gstdemux_sink_disconnect(struct strmbase_sink
*iface
)
1591 struct gstdemux
*filter
= impl_from_strmbase_sink(iface
);
1595 GST_RemoveOutputPins(filter
);
1598 static const struct strmbase_sink_ops sink_ops
=
1600 .base
.pin_query_accept
= sink_query_accept
,
1601 .sink_connect
= gstdemux_sink_connect
,
1602 .sink_disconnect
= gstdemux_sink_disconnect
,
1605 static BOOL
gstdecoder_init_gst(struct gstdemux
*filter
)
1607 GstElement
*element
= gst_element_factory_make("decodebin", NULL
);
1613 ERR("Failed to create decodebin; are %u-bit GStreamer \"base\" plugins installed?\n",
1614 8 * (int)sizeof(void*));
1618 gst_bin_add(GST_BIN(filter
->container
), element
);
1620 g_signal_connect(element
, "pad-added", G_CALLBACK(existing_new_pad_wrapper
), filter
);
1621 g_signal_connect(element
, "pad-removed", G_CALLBACK(removed_decoded_pad_wrapper
), filter
);
1622 g_signal_connect(element
, "autoplug-select", G_CALLBACK(autoplug_blacklist_wrapper
), filter
);
1623 g_signal_connect(element
, "unknown-type", G_CALLBACK(unknown_type_wrapper
), filter
);
1624 g_signal_connect(element
, "no-more-pads", G_CALLBACK(no_more_pads_wrapper
), filter
);
1626 filter
->their_sink
= gst_element_get_static_pad(element
, "sink");
1627 ResetEvent(filter
->no_more_pads_event
);
1629 if ((ret
= gst_pad_link(filter
->my_src
, filter
->their_sink
)) < 0)
1631 ERR("Failed to link pads, error %d.\n", ret
);
1635 gst_element_set_state(filter
->container
, GST_STATE_PLAYING
);
1636 ret
= gst_element_get_state(filter
->container
, NULL
, NULL
, -1);
1637 if (ret
== GST_STATE_CHANGE_FAILURE
)
1639 ERR("Failed to play stream.\n");
1643 WaitForSingleObject(filter
->no_more_pads_event
, INFINITE
);
1645 for (i
= 0; i
< filter
->source_count
; ++i
)
1647 struct gstdemux_source
*pin
= filter
->sources
[i
];
1648 const HANDLE events
[2] = {pin
->caps_event
, filter
->error_event
};
1650 pin
->seek
.llDuration
= pin
->seek
.llStop
= query_duration(pin
->their_src
);
1651 pin
->seek
.llCurrent
= 0;
1652 if (WaitForMultipleObjects(2, events
, FALSE
, INFINITE
))
1656 filter
->ignore_flush
= TRUE
;
1657 gst_element_set_state(filter
->container
, GST_STATE_READY
);
1658 gst_element_get_state(filter
->container
, NULL
, NULL
, -1);
1659 filter
->ignore_flush
= FALSE
;
1664 static HRESULT
gstdecoder_source_query_accept(struct gstdemux_source
*pin
, const AM_MEDIA_TYPE
*mt
)
1666 /* At least make sure we can convert it to GstCaps. */
1667 GstCaps
*caps
= amt_to_gst_caps(mt
);
1671 gst_caps_unref(caps
);
1675 static HRESULT
gstdecoder_source_get_media_type(struct gstdemux_source
*pin
,
1676 unsigned int index
, AM_MEDIA_TYPE
*mt
)
1678 static const GstVideoFormat video_formats
[] =
1680 /* Roughly ordered by preference from videoflip. */
1681 GST_VIDEO_FORMAT_AYUV
,
1682 GST_VIDEO_FORMAT_BGRA
,
1683 GST_VIDEO_FORMAT_BGRx
,
1684 GST_VIDEO_FORMAT_BGR
,
1685 GST_VIDEO_FORMAT_I420
,
1686 GST_VIDEO_FORMAT_YV12
,
1687 GST_VIDEO_FORMAT_YUY2
,
1688 GST_VIDEO_FORMAT_UYVY
,
1689 GST_VIDEO_FORMAT_YVYU
,
1690 GST_VIDEO_FORMAT_NV12
,
1695 CopyMediaType(mt
, &pin
->mt
);
1698 else if (IsEqualGUID(&pin
->mt
.majortype
, &MEDIATYPE_Video
)
1699 && index
- 1 < ARRAY_SIZE(video_formats
))
1701 const VIDEOINFOHEADER
*vih
= (VIDEOINFOHEADER
*)pin
->mt
.pbFormat
;
1704 gst_video_info_set_format(&info
, video_formats
[index
- 1],
1705 vih
->bmiHeader
.biWidth
, vih
->bmiHeader
.biHeight
);
1706 if (!amt_from_gst_video_info(&info
, mt
))
1707 return E_OUTOFMEMORY
;
1710 else if (IsEqualGUID(&pin
->mt
.majortype
, &MEDIATYPE_Audio
) && index
== 1)
1712 const WAVEFORMATEX
*our_format
= (WAVEFORMATEX
*)pin
->mt
.pbFormat
;
1713 WAVEFORMATEX
*format
;
1716 mt
->subtype
= MEDIASUBTYPE_PCM
;
1717 mt
->pbFormat
= CoTaskMemAlloc(sizeof(WAVEFORMATEX
));
1718 format
= (WAVEFORMATEX
*)mt
->pbFormat
;
1719 format
->wFormatTag
= WAVE_FORMAT_PCM
;
1720 format
->nChannels
= 2;
1721 format
->nSamplesPerSec
= our_format
->nSamplesPerSec
;
1722 format
->wBitsPerSample
= 16;
1723 format
->nBlockAlign
= 4;
1724 format
->nAvgBytesPerSec
= format
->nSamplesPerSec
* 4;
1729 return VFW_S_NO_MORE_ITEMS
;
1732 HRESULT
gstdemux_create(IUnknown
*outer
, IUnknown
**out
)
1734 struct gstdemux
*object
;
1736 if (!init_gstreamer())
1741 if (!(object
= heap_alloc_zero(sizeof(*object
))))
1742 return E_OUTOFMEMORY
;
1744 strmbase_filter_init(&object
->filter
, outer
, &CLSID_Gstreamer_Splitter
, &filter_ops
);
1745 strmbase_sink_init(&object
->sink
, &object
->filter
, wcsInputPinName
, &sink_ops
, NULL
);
1747 object
->no_more_pads_event
= CreateEventW(NULL
, FALSE
, FALSE
, NULL
);
1748 object
->error_event
= CreateEventW(NULL
, TRUE
, FALSE
, NULL
);
1749 object
->init_gst
= gstdecoder_init_gst
;
1750 object
->source_query_accept
= gstdecoder_source_query_accept
;
1751 object
->source_get_media_type
= gstdecoder_source_get_media_type
;
1753 TRACE("Created GStreamer demuxer %p.\n", object
);
1754 *out
= &object
->filter
.IUnknown_inner
;
1758 static struct gstdemux
*impl_from_IAMStreamSelect(IAMStreamSelect
*iface
)
1760 return CONTAINING_RECORD(iface
, struct gstdemux
, IAMStreamSelect_iface
);
1763 static HRESULT WINAPI
stream_select_QueryInterface(IAMStreamSelect
*iface
, REFIID iid
, void **out
)
1765 struct gstdemux
*filter
= impl_from_IAMStreamSelect(iface
);
1766 return IUnknown_QueryInterface(filter
->filter
.outer_unk
, iid
, out
);
1769 static ULONG WINAPI
stream_select_AddRef(IAMStreamSelect
*iface
)
1771 struct gstdemux
*filter
= impl_from_IAMStreamSelect(iface
);
1772 return IUnknown_AddRef(filter
->filter
.outer_unk
);
1775 static ULONG WINAPI
stream_select_Release(IAMStreamSelect
*iface
)
1777 struct gstdemux
*filter
= impl_from_IAMStreamSelect(iface
);
1778 return IUnknown_Release(filter
->filter
.outer_unk
);
1781 static HRESULT WINAPI
stream_select_Count(IAMStreamSelect
*iface
, DWORD
*count
)
1783 FIXME("iface %p, count %p, stub!\n", iface
, count
);
1787 static HRESULT WINAPI
stream_select_Info(IAMStreamSelect
*iface
, LONG index
,
1788 AM_MEDIA_TYPE
**mt
, DWORD
*flags
, LCID
*lcid
, DWORD
*group
, WCHAR
**name
,
1789 IUnknown
**object
, IUnknown
**unknown
)
1791 FIXME("iface %p, index %d, mt %p, flags %p, lcid %p, group %p, name %p, object %p, unknown %p, stub!\n",
1792 iface
, index
, mt
, flags
, lcid
, group
, name
, object
, unknown
);
1796 static HRESULT WINAPI
stream_select_Enable(IAMStreamSelect
*iface
, LONG index
, DWORD flags
)
1798 FIXME("iface %p, index %d, flags %#x, stub!\n", iface
, index
, flags
);
1802 static const IAMStreamSelectVtbl stream_select_vtbl
=
1804 stream_select_QueryInterface
,
1805 stream_select_AddRef
,
1806 stream_select_Release
,
1807 stream_select_Count
,
1809 stream_select_Enable
,
1812 static HRESULT WINAPI
GST_ChangeCurrent(IMediaSeeking
*iface
)
1814 struct gstdemux_source
*This
= impl_from_IMediaSeeking(iface
);
1815 TRACE("(%p)\n", This
);
1819 static HRESULT WINAPI
GST_ChangeStop(IMediaSeeking
*iface
)
1821 struct gstdemux_source
*This
= impl_from_IMediaSeeking(iface
);
1822 TRACE("(%p)\n", This
);
1826 static HRESULT WINAPI
GST_ChangeRate(IMediaSeeking
*iface
)
1828 struct gstdemux_source
*This
= impl_from_IMediaSeeking(iface
);
1829 GstEvent
*ev
= gst_event_new_seek(This
->seek
.dRate
, GST_FORMAT_TIME
, 0, GST_SEEK_TYPE_NONE
, -1, GST_SEEK_TYPE_NONE
, -1);
1830 TRACE("(%p) New rate %g\n", This
, This
->seek
.dRate
);
1832 gst_pad_push_event(This
->my_sink
, ev
);
1836 static HRESULT WINAPI
GST_Seeking_QueryInterface(IMediaSeeking
*iface
, REFIID riid
, void **ppv
)
1838 struct gstdemux_source
*This
= impl_from_IMediaSeeking(iface
);
1839 return IPin_QueryInterface(&This
->pin
.pin
.IPin_iface
, riid
, ppv
);
1842 static ULONG WINAPI
GST_Seeking_AddRef(IMediaSeeking
*iface
)
1844 struct gstdemux_source
*This
= impl_from_IMediaSeeking(iface
);
1845 return IPin_AddRef(&This
->pin
.pin
.IPin_iface
);
1848 static ULONG WINAPI
GST_Seeking_Release(IMediaSeeking
*iface
)
1850 struct gstdemux_source
*This
= impl_from_IMediaSeeking(iface
);
1851 return IPin_Release(&This
->pin
.pin
.IPin_iface
);
1854 static HRESULT WINAPI
GST_Seeking_GetCurrentPosition(IMediaSeeking
*iface
, REFERENCE_TIME
*pos
)
1856 struct gstdemux_source
*This
= impl_from_IMediaSeeking(iface
);
1858 TRACE("(%p)->(%p)\n", This
, pos
);
1865 if (This
->pin
.pin
.filter
->state
== State_Stopped
)
1867 *pos
= This
->seek
.llCurrent
;
1868 TRACE("Cached value\n");
1872 if (!gst_pad_query_position(This
->their_src
, GST_FORMAT_TIME
, pos
)) {
1873 WARN("Could not query position\n");
1877 This
->seek
.llCurrent
= *pos
;
1881 static GstSeekType
type_from_flags(DWORD flags
)
1883 switch (flags
& AM_SEEKING_PositioningBitsMask
) {
1884 case AM_SEEKING_NoPositioning
:
1885 return GST_SEEK_TYPE_NONE
;
1886 case AM_SEEKING_AbsolutePositioning
:
1887 case AM_SEEKING_RelativePositioning
:
1888 return GST_SEEK_TYPE_SET
;
1889 case AM_SEEKING_IncrementalPositioning
:
1890 return GST_SEEK_TYPE_END
;
1892 return GST_SEEK_TYPE_NONE
;
1895 static HRESULT WINAPI
GST_Seeking_SetPositions(IMediaSeeking
*iface
,
1896 REFERENCE_TIME
*pCur
, DWORD curflags
, REFERENCE_TIME
*pStop
,
1900 struct gstdemux_source
*This
= impl_from_IMediaSeeking(iface
);
1902 GstSeekType curtype
, stoptype
;
1904 gint64 stop_pos
= 0, curr_pos
= 0;
1906 TRACE("(%p)->(%p, 0x%x, %p, 0x%x)\n", This
, pCur
, curflags
, pStop
, stopflags
);
1910 hr
= SourceSeekingImpl_SetPositions(iface
, pCur
, curflags
, pStop
, stopflags
);
1911 if (This
->pin
.pin
.filter
->state
== State_Stopped
)
1914 curtype
= type_from_flags(curflags
);
1915 stoptype
= type_from_flags(stopflags
);
1916 if (curflags
& AM_SEEKING_SeekToKeyFrame
)
1917 f
|= GST_SEEK_FLAG_KEY_UNIT
;
1918 if (curflags
& AM_SEEKING_Segment
)
1919 f
|= GST_SEEK_FLAG_SEGMENT
;
1920 if (!(curflags
& AM_SEEKING_NoFlush
))
1921 f
|= GST_SEEK_FLAG_FLUSH
;
1923 if (((curflags
& AM_SEEKING_PositioningBitsMask
) == AM_SEEKING_RelativePositioning
) ||
1924 ((stopflags
& AM_SEEKING_PositioningBitsMask
) == AM_SEEKING_RelativePositioning
)) {
1926 gst_pad_query_position (This
->my_sink
, GST_FORMAT_TIME
, &tmp_pos
);
1927 if ((curflags
& AM_SEEKING_PositioningBitsMask
) == AM_SEEKING_RelativePositioning
)
1929 if ((stopflags
& AM_SEEKING_PositioningBitsMask
) == AM_SEEKING_RelativePositioning
)
1933 e
= gst_event_new_seek(This
->seek
.dRate
, GST_FORMAT_TIME
, f
, curtype
, pCur
? curr_pos
+ *pCur
* 100 : -1, stoptype
, pStop
? stop_pos
+ *pStop
* 100 : -1);
1934 if (gst_pad_push_event(This
->my_sink
, e
))
1940 static const IMediaSeekingVtbl GST_Seeking_Vtbl
=
1942 GST_Seeking_QueryInterface
,
1944 GST_Seeking_Release
,
1945 SourceSeekingImpl_GetCapabilities
,
1946 SourceSeekingImpl_CheckCapabilities
,
1947 SourceSeekingImpl_IsFormatSupported
,
1948 SourceSeekingImpl_QueryPreferredFormat
,
1949 SourceSeekingImpl_GetTimeFormat
,
1950 SourceSeekingImpl_IsUsingTimeFormat
,
1951 SourceSeekingImpl_SetTimeFormat
,
1952 SourceSeekingImpl_GetDuration
,
1953 SourceSeekingImpl_GetStopPosition
,
1954 GST_Seeking_GetCurrentPosition
,
1955 SourceSeekingImpl_ConvertTimeFormat
,
1956 GST_Seeking_SetPositions
,
1957 SourceSeekingImpl_GetPositions
,
1958 SourceSeekingImpl_GetAvailable
,
1959 SourceSeekingImpl_SetRate
,
1960 SourceSeekingImpl_GetRate
,
1961 SourceSeekingImpl_GetPreroll
1964 static inline struct gstdemux_source
*impl_from_IQualityControl( IQualityControl
*iface
)
1966 return CONTAINING_RECORD(iface
, struct gstdemux_source
, IQualityControl_iface
);
1969 static HRESULT WINAPI
GST_QualityControl_QueryInterface(IQualityControl
*iface
, REFIID riid
, void **ppv
)
1971 struct gstdemux_source
*pin
= impl_from_IQualityControl(iface
);
1972 return IPin_QueryInterface(&pin
->pin
.pin
.IPin_iface
, riid
, ppv
);
1975 static ULONG WINAPI
GST_QualityControl_AddRef(IQualityControl
*iface
)
1977 struct gstdemux_source
*pin
= impl_from_IQualityControl(iface
);
1978 return IPin_AddRef(&pin
->pin
.pin
.IPin_iface
);
1981 static ULONG WINAPI
GST_QualityControl_Release(IQualityControl
*iface
)
1983 struct gstdemux_source
*pin
= impl_from_IQualityControl(iface
);
1984 return IPin_Release(&pin
->pin
.pin
.IPin_iface
);
1987 static HRESULT WINAPI
GST_QualityControl_Notify(IQualityControl
*iface
, IBaseFilter
*sender
, Quality qm
)
1989 struct gstdemux_source
*pin
= impl_from_IQualityControl(iface
);
1990 GstQOSType type
= GST_QOS_TYPE_OVERFLOW
;
1991 GstClockTime timestamp
;
1992 GstClockTimeDiff diff
;
1995 TRACE("(%p)->(%p, { 0x%x %u %s %s })\n", pin
, sender
,
1996 qm
.Type
, qm
.Proportion
,
1997 wine_dbgstr_longlong(qm
.Late
),
1998 wine_dbgstr_longlong(qm
.TimeStamp
));
2002 /* GSTQOS_TYPE_OVERFLOW is also used for buffers that arrive on time, but
2003 * DirectShow filters might use Famine, so check that there actually is an
2005 if (qm
.Type
== Famine
&& qm
.Proportion
> 1000)
2006 type
= GST_QOS_TYPE_UNDERFLOW
;
2008 /* DirectShow filters sometimes pass negative timestamps (Audiosurf uses the
2009 * current time instead of the time of the last buffer). GstClockTime is
2010 * unsigned, so clamp it to 0. */
2011 timestamp
= max(qm
.TimeStamp
* 100, 0);
2013 /* The documentation specifies that timestamp + diff must be nonnegative. */
2014 diff
= qm
.Late
* 100;
2015 if (timestamp
< -diff
)
2018 evt
= gst_event_new_qos(type
, qm
.Proportion
/ 1000.0, diff
, timestamp
);
2021 WARN("Failed to create QOS event\n");
2022 return E_INVALIDARG
;
2025 gst_pad_push_event(pin
->my_sink
, evt
);
2030 static HRESULT WINAPI
GST_QualityControl_SetSink(IQualityControl
*iface
, IQualityControl
*tonotify
)
2032 struct gstdemux_source
*pin
= impl_from_IQualityControl(iface
);
2033 TRACE("(%p)->(%p)\n", pin
, pin
);
2038 static const IQualityControlVtbl GSTOutPin_QualityControl_Vtbl
= {
2039 GST_QualityControl_QueryInterface
,
2040 GST_QualityControl_AddRef
,
2041 GST_QualityControl_Release
,
2042 GST_QualityControl_Notify
,
2043 GST_QualityControl_SetSink
2046 static inline struct gstdemux_source
*impl_source_from_IPin(IPin
*iface
)
2048 return CONTAINING_RECORD(iface
, struct gstdemux_source
, pin
.pin
.IPin_iface
);
2051 static HRESULT
source_query_interface(struct strmbase_pin
*iface
, REFIID iid
, void **out
)
2053 struct gstdemux_source
*pin
= impl_source_from_IPin(&iface
->IPin_iface
);
2055 if (IsEqualGUID(iid
, &IID_IMediaSeeking
))
2056 *out
= &pin
->seek
.IMediaSeeking_iface
;
2057 else if (IsEqualGUID(iid
, &IID_IQualityControl
))
2058 *out
= &pin
->IQualityControl_iface
;
2060 return E_NOINTERFACE
;
2062 IUnknown_AddRef((IUnknown
*)*out
);
2066 static HRESULT
source_query_accept(struct strmbase_pin
*iface
, const AM_MEDIA_TYPE
*mt
)
2068 struct gstdemux_source
*pin
= impl_source_from_IPin(&iface
->IPin_iface
);
2069 struct gstdemux
*filter
= impl_from_strmbase_filter(iface
->filter
);
2070 return filter
->source_query_accept(pin
, mt
);
2073 static HRESULT
source_get_media_type(struct strmbase_pin
*iface
, unsigned int index
, AM_MEDIA_TYPE
*mt
)
2075 struct gstdemux_source
*pin
= impl_source_from_IPin(&iface
->IPin_iface
);
2076 struct gstdemux
*filter
= impl_from_strmbase_filter(iface
->filter
);
2077 return filter
->source_get_media_type(pin
, index
, mt
);
2080 static HRESULT WINAPI
GSTOutPin_DecideBufferSize(struct strmbase_source
*iface
,
2081 IMemAllocator
*allocator
, ALLOCATOR_PROPERTIES
*props
)
2083 struct gstdemux_source
*pin
= impl_source_from_IPin(&iface
->pin
.IPin_iface
);
2084 unsigned int buffer_size
= 16384;
2085 ALLOCATOR_PROPERTIES ret_props
;
2087 if (IsEqualGUID(&pin
->pin
.pin
.mt
.formattype
, &FORMAT_VideoInfo
))
2089 VIDEOINFOHEADER
*format
= (VIDEOINFOHEADER
*)pin
->pin
.pin
.mt
.pbFormat
;
2090 buffer_size
= format
->bmiHeader
.biSizeImage
;
2092 gst_util_set_object_arg(G_OBJECT(pin
->flip
), "method",
2093 format
->bmiHeader
.biCompression
== BI_RGB
? "vertical-flip" : "none");
2095 else if (IsEqualGUID(&pin
->pin
.pin
.mt
.formattype
, &FORMAT_WaveFormatEx
)
2096 && (IsEqualGUID(&pin
->pin
.pin
.mt
.subtype
, &MEDIASUBTYPE_PCM
)
2097 || IsEqualGUID(&pin
->pin
.pin
.mt
.subtype
, &MEDIASUBTYPE_IEEE_FLOAT
)))
2099 WAVEFORMATEX
*format
= (WAVEFORMATEX
*)pin
->pin
.pin
.mt
.pbFormat
;
2100 buffer_size
= format
->nAvgBytesPerSec
;
2103 props
->cBuffers
= max(props
->cBuffers
, 1);
2104 props
->cbBuffer
= max(props
->cbBuffer
, buffer_size
);
2105 props
->cbAlign
= max(props
->cbAlign
, 1);
2106 return IMemAllocator_SetProperties(allocator
, props
, &ret_props
);
2109 static void free_source_pin(struct gstdemux_source
*pin
)
2111 if (pin
->pin
.pin
.peer
)
2113 if (SUCCEEDED(IMemAllocator_Decommit(pin
->pin
.pAllocator
)))
2114 IPin_Disconnect(pin
->pin
.pin
.peer
);
2115 IPin_Disconnect(&pin
->pin
.pin
.IPin_iface
);
2122 gst_pad_unlink(pin
->their_src
, pin
->post_sink
);
2123 gst_pad_unlink(pin
->post_src
, pin
->my_sink
);
2124 gst_object_unref(pin
->post_src
);
2125 gst_object_unref(pin
->post_sink
);
2126 pin
->post_src
= pin
->post_sink
= NULL
;
2129 gst_pad_unlink(pin
->their_src
, pin
->my_sink
);
2130 gst_object_unref(pin
->their_src
);
2132 gst_object_unref(pin
->my_sink
);
2133 CloseHandle(pin
->caps_event
);
2134 CloseHandle(pin
->eos_event
);
2135 FreeMediaType(&pin
->mt
);
2136 gst_segment_free(pin
->segment
);
2138 strmbase_seeking_cleanup(&pin
->seek
);
2139 strmbase_source_cleanup(&pin
->pin
);
2143 static const struct strmbase_source_ops source_ops
=
2145 .base
.pin_query_interface
= source_query_interface
,
2146 .base
.pin_query_accept
= source_query_accept
,
2147 .base
.pin_get_media_type
= source_get_media_type
,
2148 .pfnAttemptConnection
= BaseOutputPinImpl_AttemptConnection
,
2149 .pfnDecideAllocator
= BaseOutputPinImpl_DecideAllocator
,
2150 .pfnDecideBufferSize
= GSTOutPin_DecideBufferSize
,
2153 static struct gstdemux_source
*create_pin(struct gstdemux
*filter
, const WCHAR
*name
)
2155 struct gstdemux_source
*pin
, **new_array
;
2158 if (!(new_array
= heap_realloc(filter
->sources
, (filter
->source_count
+ 1) * sizeof(*new_array
))))
2160 filter
->sources
= new_array
;
2162 if (!(pin
= heap_alloc_zero(sizeof(*pin
))))
2165 strmbase_source_init(&pin
->pin
, &filter
->filter
, name
, &source_ops
);
2166 pin
->caps_event
= CreateEventW(NULL
, FALSE
, FALSE
, NULL
);
2167 pin
->eos_event
= CreateEventW(NULL
, FALSE
, FALSE
, NULL
);
2168 pin
->segment
= gst_segment_new();
2169 gst_segment_init(pin
->segment
, GST_FORMAT_TIME
);
2170 pin
->IQualityControl_iface
.lpVtbl
= &GSTOutPin_QualityControl_Vtbl
;
2171 strmbase_seeking_init(&pin
->seek
, &GST_Seeking_Vtbl
, GST_ChangeStop
,
2172 GST_ChangeCurrent
, GST_ChangeRate
);
2173 BaseFilterImpl_IncrementPinVersion(&filter
->filter
);
2175 sprintf(pad_name
, "qz_sink_%u", filter
->source_count
);
2176 pin
->my_sink
= gst_pad_new(pad_name
, GST_PAD_SINK
);
2177 gst_pad_set_element_private(pin
->my_sink
, pin
);
2178 gst_pad_set_chain_function(pin
->my_sink
, got_data_sink_wrapper
);
2179 gst_pad_set_event_function(pin
->my_sink
, event_sink_wrapper
);
2180 gst_pad_set_query_function(pin
->my_sink
, query_sink_wrapper
);
2182 filter
->sources
[filter
->source_count
++] = pin
;
2186 static HRESULT
GST_RemoveOutputPins(struct gstdemux
*This
)
2190 TRACE("(%p)\n", This
);
2193 if (!This
->container
)
2195 gst_element_set_state(This
->container
, GST_STATE_NULL
);
2196 gst_pad_unlink(This
->my_src
, This
->their_sink
);
2197 gst_object_unref(This
->my_src
);
2198 gst_object_unref(This
->their_sink
);
2199 This
->my_src
= This
->their_sink
= NULL
;
2201 for (i
= 0; i
< This
->source_count
; ++i
)
2202 free_source_pin(This
->sources
[i
]);
2204 This
->source_count
= 0;
2205 heap_free(This
->sources
);
2206 This
->sources
= NULL
;
2207 gst_element_set_bus(This
->container
, NULL
);
2208 gst_object_unref(This
->container
);
2209 This
->container
= NULL
;
2210 BaseFilterImpl_IncrementPinVersion(&This
->filter
);
2214 void perform_cb_gstdemux(struct cb_data
*cbdata
)
2216 switch(cbdata
->type
)
2220 struct watch_bus_data
*data
= &cbdata
->u
.watch_bus_data
;
2221 cbdata
->u
.watch_bus_data
.ret
= watch_bus(data
->bus
, data
->msg
, data
->user
);
2224 case EXISTING_NEW_PAD
:
2226 struct pad_added_data
*data
= &cbdata
->u
.pad_added_data
;
2227 existing_new_pad(data
->element
, data
->pad
, data
->user
);
2230 case QUERY_FUNCTION
:
2232 struct query_function_data
*data
= &cbdata
->u
.query_function_data
;
2233 cbdata
->u
.query_function_data
.ret
= query_function(data
->pad
, data
->parent
, data
->query
);
2238 struct activate_mode_data
*data
= &cbdata
->u
.activate_mode_data
;
2239 cbdata
->u
.activate_mode_data
.ret
= activate_mode(data
->pad
, data
->parent
, data
->mode
, data
->activate
);
2244 struct no_more_pads_data
*data
= &cbdata
->u
.no_more_pads_data
;
2245 no_more_pads(data
->element
, data
->user
);
2248 case REQUEST_BUFFER_SRC
:
2250 struct getrange_data
*data
= &cbdata
->u
.getrange_data
;
2251 cbdata
->u
.getrange_data
.ret
= request_buffer_src(data
->pad
, data
->parent
,
2252 data
->ofs
, data
->len
, data
->buf
);
2257 struct event_src_data
*data
= &cbdata
->u
.event_src_data
;
2258 cbdata
->u
.event_src_data
.ret
= event_src(data
->pad
, data
->parent
, data
->event
);
2263 struct event_sink_data
*data
= &cbdata
->u
.event_sink_data
;
2264 cbdata
->u
.event_sink_data
.ret
= event_sink(data
->pad
, data
->parent
, data
->event
);
2269 struct got_data_sink_data
*data
= &cbdata
->u
.got_data_sink_data
;
2270 cbdata
->u
.got_data_sink_data
.ret
= got_data_sink(data
->pad
, data
->parent
, data
->buf
);
2273 case REMOVED_DECODED_PAD
:
2275 struct pad_removed_data
*data
= &cbdata
->u
.pad_removed_data
;
2276 removed_decoded_pad(data
->element
, data
->pad
, data
->user
);
2279 case AUTOPLUG_BLACKLIST
:
2281 struct autoplug_blacklist_data
*data
= &cbdata
->u
.autoplug_blacklist_data
;
2282 cbdata
->u
.autoplug_blacklist_data
.ret
= autoplug_blacklist(data
->bin
,
2283 data
->pad
, data
->caps
, data
->fact
, data
->user
);
2288 struct unknown_type_data
*data
= &cbdata
->u
.unknown_type_data
;
2289 unknown_type(data
->bin
, data
->pad
, data
->caps
, data
->user
);
2294 struct query_sink_data
*data
= &cbdata
->u
.query_sink_data
;
2295 cbdata
->u
.query_sink_data
.ret
= query_sink(data
->pad
, data
->parent
,
2306 static BOOL
compare_media_types(const AM_MEDIA_TYPE
*a
, const AM_MEDIA_TYPE
*b
)
2308 return IsEqualGUID(&a
->majortype
, &b
->majortype
)
2309 && IsEqualGUID(&a
->subtype
, &b
->subtype
)
2310 && IsEqualGUID(&a
->formattype
, &b
->formattype
)
2311 && a
->cbFormat
== b
->cbFormat
2312 && !memcmp(a
->pbFormat
, b
->pbFormat
, a
->cbFormat
);
2315 static HRESULT
wave_parser_sink_query_accept(struct strmbase_pin
*iface
, const AM_MEDIA_TYPE
*mt
)
2317 if (!IsEqualGUID(&mt
->majortype
, &MEDIATYPE_Stream
))
2319 if (IsEqualGUID(&mt
->subtype
, &MEDIASUBTYPE_WAVE
))
2321 if (IsEqualGUID(&mt
->subtype
, &MEDIASUBTYPE_AU
) || IsEqualGUID(&mt
->subtype
, &MEDIASUBTYPE_AIFF
))
2322 FIXME("AU and AIFF files are not yet supported.\n");
2326 static const struct strmbase_sink_ops wave_parser_sink_ops
=
2328 .base
.pin_query_accept
= wave_parser_sink_query_accept
,
2329 .sink_connect
= gstdemux_sink_connect
,
2330 .sink_disconnect
= gstdemux_sink_disconnect
,
2333 static BOOL
wave_parser_init_gst(struct gstdemux
*filter
)
2335 static const WCHAR source_name
[] = {'o','u','t','p','u','t',0};
2336 struct gstdemux_source
*pin
;
2337 GstElement
*element
;
2341 if (!(element
= gst_element_factory_make("wavparse", NULL
)))
2343 ERR("Failed to create wavparse; are %u-bit GStreamer \"good\" plugins installed?\n",
2344 8 * (int)sizeof(void*));
2348 gst_bin_add(GST_BIN(filter
->container
), element
);
2350 filter
->their_sink
= gst_element_get_static_pad(element
, "sink");
2351 if ((ret
= gst_pad_link(filter
->my_src
, filter
->their_sink
)) < 0)
2353 ERR("Failed to link sink pads, error %d.\n", ret
);
2357 if (!(pin
= create_pin(filter
, source_name
)))
2359 pin
->their_src
= gst_element_get_static_pad(element
, "src");
2360 gst_object_ref(pin
->their_src
);
2361 if ((ret
= gst_pad_link(pin
->their_src
, pin
->my_sink
)) < 0)
2363 ERR("Failed to link source pads, error %d.\n", ret
);
2367 gst_pad_set_active(pin
->my_sink
, 1);
2368 gst_element_set_state(filter
->container
, GST_STATE_PAUSED
);
2369 ret
= gst_element_get_state(filter
->container
, NULL
, NULL
, -1);
2370 if (ret
== GST_STATE_CHANGE_FAILURE
)
2372 ERR("Failed to play stream.\n");
2376 pin
->seek
.llDuration
= pin
->seek
.llStop
= query_duration(pin
->their_src
);
2377 pin
->seek
.llCurrent
= 0;
2379 events
[0] = pin
->caps_event
;
2380 events
[1] = filter
->error_event
;
2381 if (WaitForMultipleObjects(2, events
, FALSE
, INFINITE
))
2384 filter
->ignore_flush
= TRUE
;
2385 gst_element_set_state(filter
->container
, GST_STATE_READY
);
2386 gst_element_get_state(filter
->container
, NULL
, NULL
, -1);
2387 filter
->ignore_flush
= FALSE
;
2392 static HRESULT
wave_parser_source_query_accept(struct gstdemux_source
*pin
, const AM_MEDIA_TYPE
*mt
)
2394 return compare_media_types(mt
, &pin
->mt
) ? S_OK
: S_FALSE
;
2397 static HRESULT
wave_parser_source_get_media_type(struct gstdemux_source
*pin
,
2398 unsigned int index
, AM_MEDIA_TYPE
*mt
)
2401 return VFW_S_NO_MORE_ITEMS
;
2402 CopyMediaType(mt
, &pin
->mt
);
2406 HRESULT
wave_parser_create(IUnknown
*outer
, IUnknown
**out
)
2408 static const WCHAR sink_name
[] = {'i','n','p','u','t',' ','p','i','n',0};
2409 struct gstdemux
*object
;
2411 if (!init_gstreamer())
2416 if (!(object
= heap_alloc_zero(sizeof(*object
))))
2417 return E_OUTOFMEMORY
;
2419 strmbase_filter_init(&object
->filter
, outer
, &CLSID_WAVEParser
, &filter_ops
);
2420 strmbase_sink_init(&object
->sink
, &object
->filter
, sink_name
, &wave_parser_sink_ops
, NULL
);
2421 object
->init_gst
= wave_parser_init_gst
;
2422 object
->error_event
= CreateEventW(NULL
, TRUE
, FALSE
, NULL
);
2423 object
->source_query_accept
= wave_parser_source_query_accept
;
2424 object
->source_get_media_type
= wave_parser_source_get_media_type
;
2426 TRACE("Created WAVE parser %p.\n", object
);
2427 *out
= &object
->filter
.IUnknown_inner
;
2431 static HRESULT
avi_splitter_sink_query_accept(struct strmbase_pin
*iface
, const AM_MEDIA_TYPE
*mt
)
2433 if (IsEqualGUID(&mt
->majortype
, &MEDIATYPE_Stream
)
2434 && IsEqualGUID(&mt
->subtype
, &MEDIASUBTYPE_Avi
))
2439 static const struct strmbase_sink_ops avi_splitter_sink_ops
=
2441 .base
.pin_query_accept
= avi_splitter_sink_query_accept
,
2442 .sink_connect
= gstdemux_sink_connect
,
2443 .sink_disconnect
= gstdemux_sink_disconnect
,
2446 static BOOL
avi_splitter_init_gst(struct gstdemux
*filter
)
2448 GstElement
*element
= gst_element_factory_make("avidemux", NULL
);
2454 ERR("Failed to create avidemux; are %u-bit GStreamer \"good\" plugins installed?\n",
2455 8 * (int)sizeof(void*));
2459 gst_bin_add(GST_BIN(filter
->container
), element
);
2461 g_signal_connect(element
, "pad-added", G_CALLBACK(existing_new_pad_wrapper
), filter
);
2462 g_signal_connect(element
, "pad-removed", G_CALLBACK(removed_decoded_pad_wrapper
), filter
);
2463 g_signal_connect(element
, "no-more-pads", G_CALLBACK(no_more_pads_wrapper
), filter
);
2465 filter
->their_sink
= gst_element_get_static_pad(element
, "sink");
2466 ResetEvent(filter
->no_more_pads_event
);
2468 if ((ret
= gst_pad_link(filter
->my_src
, filter
->their_sink
)) < 0)
2470 ERR("Failed to link pads, error %d.\n", ret
);
2474 gst_element_set_state(filter
->container
, GST_STATE_PLAYING
);
2475 ret
= gst_element_get_state(filter
->container
, NULL
, NULL
, -1);
2476 if (ret
== GST_STATE_CHANGE_FAILURE
)
2478 ERR("Failed to play stream.\n");
2482 WaitForSingleObject(filter
->no_more_pads_event
, INFINITE
);
2484 for (i
= 0; i
< filter
->source_count
; ++i
)
2486 struct gstdemux_source
*pin
= filter
->sources
[i
];
2487 const HANDLE events
[2] = {pin
->caps_event
, filter
->error_event
};
2489 pin
->seek
.llDuration
= pin
->seek
.llStop
= query_duration(pin
->their_src
);
2490 pin
->seek
.llCurrent
= 0;
2491 if (WaitForMultipleObjects(2, events
, FALSE
, INFINITE
))
2495 filter
->ignore_flush
= TRUE
;
2496 gst_element_set_state(filter
->container
, GST_STATE_READY
);
2497 gst_element_get_state(filter
->container
, NULL
, NULL
, -1);
2498 filter
->ignore_flush
= FALSE
;
2503 static HRESULT
avi_splitter_source_query_accept(struct gstdemux_source
*pin
, const AM_MEDIA_TYPE
*mt
)
2505 return compare_media_types(mt
, &pin
->mt
) ? S_OK
: S_FALSE
;
2508 static HRESULT
avi_splitter_source_get_media_type(struct gstdemux_source
*pin
,
2509 unsigned int index
, AM_MEDIA_TYPE
*mt
)
2512 return VFW_S_NO_MORE_ITEMS
;
2513 CopyMediaType(mt
, &pin
->mt
);
2517 HRESULT
avi_splitter_create(IUnknown
*outer
, IUnknown
**out
)
2519 static const WCHAR sink_name
[] = {'i','n','p','u','t',' ','p','i','n',0};
2520 struct gstdemux
*object
;
2522 if (!init_gstreamer())
2527 if (!(object
= heap_alloc_zero(sizeof(*object
))))
2528 return E_OUTOFMEMORY
;
2530 strmbase_filter_init(&object
->filter
, outer
, &CLSID_AviSplitter
, &filter_ops
);
2531 strmbase_sink_init(&object
->sink
, &object
->filter
, sink_name
, &avi_splitter_sink_ops
, NULL
);
2532 object
->no_more_pads_event
= CreateEventW(NULL
, FALSE
, FALSE
, NULL
);
2533 object
->error_event
= CreateEventW(NULL
, TRUE
, FALSE
, NULL
);
2534 object
->init_gst
= avi_splitter_init_gst
;
2535 object
->source_query_accept
= avi_splitter_source_query_accept
;
2536 object
->source_get_media_type
= avi_splitter_source_get_media_type
;
2538 TRACE("Created AVI splitter %p.\n", object
);
2539 *out
= &object
->filter
.IUnknown_inner
;
2543 static HRESULT
mpeg_splitter_sink_query_accept(struct strmbase_pin
*iface
, const AM_MEDIA_TYPE
*mt
)
2545 if (!IsEqualGUID(&mt
->majortype
, &MEDIATYPE_Stream
))
2547 if (IsEqualGUID(&mt
->subtype
, &MEDIASUBTYPE_MPEG1Audio
))
2549 if (IsEqualGUID(&mt
->subtype
, &MEDIASUBTYPE_MPEG1Video
)
2550 || IsEqualGUID(&mt
->subtype
, &MEDIASUBTYPE_MPEG1System
)
2551 || IsEqualGUID(&mt
->subtype
, &MEDIASUBTYPE_MPEG1VideoCD
))
2552 FIXME("Unsupported subtype %s.\n", wine_dbgstr_guid(&mt
->subtype
));
2556 static const struct strmbase_sink_ops mpeg_splitter_sink_ops
=
2558 .base
.pin_query_accept
= mpeg_splitter_sink_query_accept
,
2559 .sink_connect
= gstdemux_sink_connect
,
2560 .sink_disconnect
= gstdemux_sink_disconnect
,
2563 static BOOL
mpeg_splitter_init_gst(struct gstdemux
*filter
)
2565 static const WCHAR source_name
[] = {'A','u','d','i','o',0};
2566 struct gstdemux_source
*pin
;
2567 GstElement
*element
;
2572 if (!(element
= gst_element_factory_make("mpegaudioparse", NULL
)))
2574 ERR("Failed to create mpegaudioparse; are %u-bit GStreamer \"good\" plugins installed?\n",
2575 8 * (int)sizeof(void*));
2579 gst_bin_add(GST_BIN(filter
->container
), element
);
2581 filter
->their_sink
= gst_element_get_static_pad(element
, "sink");
2582 if ((ret
= gst_pad_link(filter
->my_src
, filter
->their_sink
)) < 0)
2584 ERR("Failed to link sink pads, error %d.\n", ret
);
2588 if (!(pin
= create_pin(filter
, source_name
)))
2590 gst_object_ref(pin
->their_src
= gst_element_get_static_pad(element
, "src"));
2591 if ((ret
= gst_pad_link(pin
->their_src
, pin
->my_sink
)) < 0)
2593 ERR("Failed to link source pads, error %d.\n", ret
);
2597 gst_pad_set_active(pin
->my_sink
, 1);
2598 gst_element_set_state(filter
->container
, GST_STATE_PAUSED
);
2599 ret
= gst_element_get_state(filter
->container
, NULL
, NULL
, -1);
2600 if (ret
== GST_STATE_CHANGE_FAILURE
)
2602 ERR("Failed to play stream.\n");
2606 events
[0] = filter
->duration_event
;
2607 events
[1] = filter
->error_event
;
2608 events
[2] = pin
->eos_event
;
2609 res
= WaitForMultipleObjects(3, events
, FALSE
, INFINITE
);
2613 pin
->seek
.llDuration
= pin
->seek
.llStop
= query_duration(pin
->their_src
);
2614 pin
->seek
.llCurrent
= 0;
2616 events
[0] = pin
->caps_event
;
2617 if (WaitForMultipleObjects(2, events
, FALSE
, INFINITE
))
2620 filter
->ignore_flush
= TRUE
;
2621 gst_element_set_state(filter
->container
, GST_STATE_READY
);
2622 gst_element_get_state(filter
->container
, NULL
, NULL
, -1);
2623 filter
->ignore_flush
= FALSE
;
2628 static HRESULT
mpeg_splitter_source_query_accept(struct gstdemux_source
*pin
, const AM_MEDIA_TYPE
*mt
)
2630 return compare_media_types(mt
, &pin
->mt
) ? S_OK
: S_FALSE
;
2633 static HRESULT
mpeg_splitter_source_get_media_type(struct gstdemux_source
*pin
,
2634 unsigned int index
, AM_MEDIA_TYPE
*mt
)
2637 return VFW_S_NO_MORE_ITEMS
;
2638 CopyMediaType(mt
, &pin
->mt
);
2642 static HRESULT
mpeg_splitter_query_interface(struct strmbase_filter
*iface
, REFIID iid
, void **out
)
2644 struct gstdemux
*filter
= impl_from_strmbase_filter(iface
);
2646 if (IsEqualGUID(iid
, &IID_IAMStreamSelect
))
2648 *out
= &filter
->IAMStreamSelect_iface
;
2649 IUnknown_AddRef((IUnknown
*)*out
);
2653 return E_NOINTERFACE
;
2656 static const struct strmbase_filter_ops mpeg_splitter_ops
=
2658 .filter_query_interface
= mpeg_splitter_query_interface
,
2659 .filter_get_pin
= gstdemux_get_pin
,
2660 .filter_destroy
= gstdemux_destroy
,
2661 .filter_init_stream
= gstdemux_init_stream
,
2662 .filter_start_stream
= gstdemux_start_stream
,
2663 .filter_stop_stream
= gstdemux_stop_stream
,
2664 .filter_cleanup_stream
= gstdemux_cleanup_stream
,
2665 .filter_wait_state
= gstdemux_wait_state
,
2668 HRESULT
mpeg_splitter_create(IUnknown
*outer
, IUnknown
**out
)
2670 static const WCHAR sink_name
[] = {'I','n','p','u','t',0};
2671 struct gstdemux
*object
;
2673 if (!init_gstreamer())
2678 if (!(object
= heap_alloc_zero(sizeof(*object
))))
2679 return E_OUTOFMEMORY
;
2681 strmbase_filter_init(&object
->filter
, outer
, &CLSID_MPEG1Splitter
, &mpeg_splitter_ops
);
2682 strmbase_sink_init(&object
->sink
, &object
->filter
, sink_name
, &mpeg_splitter_sink_ops
, NULL
);
2683 object
->IAMStreamSelect_iface
.lpVtbl
= &stream_select_vtbl
;
2685 object
->duration_event
= CreateEventW(NULL
, FALSE
, FALSE
, NULL
);
2686 object
->error_event
= CreateEventW(NULL
, TRUE
, FALSE
, NULL
);
2687 object
->init_gst
= mpeg_splitter_init_gst
;
2688 object
->source_query_accept
= mpeg_splitter_source_query_accept
;
2689 object
->source_get_media_type
= mpeg_splitter_source_get_media_type
;
2690 object
->enum_sink_first
= TRUE
;
2692 TRACE("Created MPEG-1 splitter %p.\n", object
);
2693 *out
= &object
->filter
.IUnknown_inner
;