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/app/gstappsink.h>
24 #include <gst/app/gstappsrc.h>
25 #include <gst/app/gstappbuffer.h>
26 #include <gst/gstutils.h>
28 #include "gst_private.h"
29 #include "gst_guids.h"
34 #include "wine/unicode.h"
35 #include "wine/debug.h"
45 WINE_DEFAULT_DEBUG_CHANNEL(gstreamer
);
47 typedef struct GSTOutPin GSTOutPin
;
48 typedef struct GSTInPin
{
50 IAsyncReader
*pReader
;
51 IMemAllocator
*pAlloc
;
54 typedef struct GSTImpl
{
63 BOOL discont
, initial
;
64 GstElement
*gstfilter
;
65 GstPad
*my_src
, *their_sink
;
67 guint64 start
, nextofs
, nextpullofs
, stop
;
68 ALLOCATOR_PROPERTIES props
;
69 HANDLE event
, changed_ofs
;
76 IQualityControl IQualityControl_iface
;
87 static const WCHAR wcsInputPinName
[] = {'i','n','p','u','t',' ','p','i','n',0};
88 static const IMediaSeekingVtbl GST_Seeking_Vtbl
;
89 static const IPinVtbl GST_OutputPin_Vtbl
;
90 static const IPinVtbl GST_InputPin_Vtbl
;
91 static const IBaseFilterVtbl GST_Vtbl
;
92 static const IQualityControlVtbl GSTOutPin_QualityControl_Vtbl
;
94 static HRESULT
GST_AddPin(GSTImpl
*This
, const PIN_INFO
*piOutput
, const AM_MEDIA_TYPE
*amt
);
95 static HRESULT
GST_RemoveOutputPins(GSTImpl
*This
);
96 static HRESULT WINAPI
GST_ChangeCurrent(IMediaSeeking
*iface
);
97 static HRESULT WINAPI
GST_ChangeStop(IMediaSeeking
*iface
);
98 static HRESULT WINAPI
GST_ChangeRate(IMediaSeeking
*iface
);
100 static gboolean
amt_from_gst_caps_audio(GstCaps
*caps
, AM_MEDIA_TYPE
*amt
) {
101 WAVEFORMATEXTENSIBLE
*wfe
;
104 gint32 depth
= 0, bpp
= 0;
105 const char *typename
;
106 arg
= gst_caps_get_structure(caps
, 0);
107 typename
= gst_structure_get_name(arg
);
111 wfe
= CoTaskMemAlloc(sizeof(*wfe
));
112 wfx
= (WAVEFORMATEX
*)wfe
;
113 amt
->majortype
= MEDIATYPE_Audio
;
114 amt
->subtype
= MEDIASUBTYPE_PCM
;
115 amt
->formattype
= FORMAT_WaveFormatEx
;
116 amt
->pbFormat
= (BYTE
*)wfe
;
117 amt
->cbFormat
= sizeof(*wfe
);
118 amt
->bFixedSizeSamples
= 0;
119 amt
->bTemporalCompression
= 1;
120 amt
->lSampleSize
= 0;
123 wfx
->wFormatTag
= WAVE_FORMAT_EXTENSIBLE
;
124 if (!gst_structure_get_int(arg
, "channels", (INT
*)&wfx
->nChannels
))
126 if (!gst_structure_get_int(arg
, "rate", (INT
*)&wfx
->nSamplesPerSec
))
128 gst_structure_get_int(arg
, "width", &depth
);
129 gst_structure_get_int(arg
, "depth", &bpp
);
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 (!strcmp(typename
, "audio/x-raw-float")) {
148 wfe
->SubFormat
= KSDATAFORMAT_SUBTYPE_IEEE_FLOAT
;
149 wfx
->wBitsPerSample
= wfe
->Samples
.wValidBitsPerSample
= 32;
151 wfe
->SubFormat
= KSDATAFORMAT_SUBTYPE_PCM
;
152 if (wfx
->nChannels
<= 2 && bpp
<= 16 && depth
== bpp
) {
153 wfx
->wFormatTag
= WAVE_FORMAT_PCM
;
157 wfx
->nBlockAlign
= wfx
->nChannels
* wfx
->wBitsPerSample
/8;
158 wfx
->nAvgBytesPerSec
= wfx
->nSamplesPerSec
* wfx
->nBlockAlign
;
162 static gboolean
amt_from_gst_caps_video(GstCaps
*caps
, AM_MEDIA_TYPE
*amt
) {
163 VIDEOINFOHEADER
*vih
= CoTaskMemAlloc(sizeof(*vih
));
164 BITMAPINFOHEADER
*bih
= &vih
->bmiHeader
;
166 gint32 width
= 0, height
= 0, nom
= 0, denom
= 0;
167 const char *typename
;
168 arg
= gst_caps_get_structure(caps
, 0);
169 typename
= gst_structure_get_name(arg
);
172 if (!gst_structure_get_int(arg
, "width", &width
) ||
173 !gst_structure_get_int(arg
, "height", &height
) ||
174 !gst_structure_get_fraction(arg
, "framerate", &nom
, &denom
))
176 amt
->formattype
= FORMAT_VideoInfo
;
177 amt
->pbFormat
= (BYTE
*)vih
;
178 amt
->cbFormat
= sizeof(*vih
);
179 amt
->bFixedSizeSamples
= amt
->bTemporalCompression
= 1;
180 amt
->lSampleSize
= 0;
182 ZeroMemory(vih
, sizeof(*vih
));
183 amt
->majortype
= MEDIATYPE_Video
;
184 if (!strcmp(typename
, "video/x-raw-rgb")) {
185 if (!gst_structure_get_int(arg
, "bpp", (INT
*)&bih
->biBitCount
))
187 switch (bih
->biBitCount
) {
188 case 16: amt
->subtype
= MEDIASUBTYPE_RGB555
; break;
189 case 24: amt
->subtype
= MEDIASUBTYPE_RGB24
; break;
190 case 32: amt
->subtype
= MEDIASUBTYPE_RGB32
; break;
192 FIXME("Unknown bpp %u\n", bih
->biBitCount
);
195 bih
->biCompression
= BI_RGB
;
197 amt
->subtype
= MEDIATYPE_Video
;
198 if (!gst_structure_get_fourcc(arg
, "format", &amt
->subtype
.Data1
))
200 switch (amt
->subtype
.Data1
) {
201 case mmioFOURCC('I','4','2','0'):
202 case mmioFOURCC('Y','V','1','2'):
203 case mmioFOURCC('N','V','1','2'):
204 case mmioFOURCC('N','V','2','1'):
205 bih
->biBitCount
= 12; break;
206 case mmioFOURCC('Y','U','Y','2'):
207 case mmioFOURCC('Y','V','Y','U'):
208 bih
->biBitCount
= 16; break;
210 bih
->biCompression
= amt
->subtype
.Data1
;
212 bih
->biSizeImage
= width
* height
* bih
->biBitCount
/ 8;
213 vih
->AvgTimePerFrame
= 10000000;
214 vih
->AvgTimePerFrame
*= denom
;
215 vih
->AvgTimePerFrame
/= nom
;
216 vih
->rcSource
.left
= 0;
217 vih
->rcSource
.right
= width
;
218 vih
->rcSource
.top
= height
;
219 vih
->rcSource
.bottom
= 0;
220 vih
->rcTarget
= vih
->rcSource
;
221 bih
->biSize
= sizeof(*bih
);
222 bih
->biWidth
= width
;
223 bih
->biHeight
= height
;
228 static gboolean
accept_caps_sink(GstPad
*pad
, GstCaps
*caps
) {
229 GSTOutPin
*pin
= gst_pad_get_element_private(pad
);
232 const char *typename
;
234 arg
= gst_caps_get_structure(caps
, 0);
235 typename
= gst_structure_get_name(arg
);
236 if (!strcmp(typename
, "audio/x-raw-int") ||
237 !strcmp(typename
, "audio/x-raw-float")) {
239 ERR("Setting audio caps on non-audio pad?\n");
242 ret
= amt_from_gst_caps_audio(caps
, &amt
);
246 } else if (!strcmp(typename
, "video/x-raw-rgb")
247 || !strcmp(typename
, "video/x-raw-yuv")) {
249 ERR("Setting video caps on non-video pad?\n");
252 ret
= amt_from_gst_caps_video(caps
, &amt
);
257 FIXME("Unhandled type \"%s\"\n", typename
);
262 static gboolean
setcaps_sink(GstPad
*pad
, GstCaps
*caps
) {
263 GSTOutPin
*pin
= gst_pad_get_element_private(pad
);
264 GSTImpl
*This
= (GSTImpl
*)pin
->pin
.pin
.pinInfo
.pFilter
;
267 const char *typename
;
269 arg
= gst_caps_get_structure(caps
, 0);
270 typename
= gst_structure_get_name(arg
);
271 if (!strcmp(typename
, "audio/x-raw-int") ||
272 !strcmp(typename
, "audio/x-raw-float")) {
274 ERR("Setting audio caps on non-audio pad?\n");
277 ret
= amt_from_gst_caps_audio(caps
, &amt
);
278 } else if (!strcmp(typename
, "video/x-raw-rgb")
279 || !strcmp(typename
, "video/x-raw-yuv")) {
281 ERR("Setting video caps on non-video pad?\n");
284 ret
= amt_from_gst_caps_video(caps
, &amt
);
286 This
->props
.cbBuffer
= max(This
->props
.cbBuffer
, ((VIDEOINFOHEADER
*)amt
.pbFormat
)->bmiHeader
.biSizeImage
);
288 FIXME("Unhandled type \"%s\"\n", typename
);
291 TRACE("Linking returned %i for %s\n", ret
, typename
);
294 FreeMediaType(pin
->pmt
);
299 static gboolean
gst_base_src_perform_seek(GSTImpl
*This
, GstEvent
*event
)
303 GstFormat seek_format
;
305 GstSeekType cur_type
, stop_type
;
310 BOOL thread
= !!This
->push_thread
;
312 gst_event_parse_seek(event
, &rate
, &seek_format
, &flags
,
313 &cur_type
, &cur
, &stop_type
, &stop
);
315 if (seek_format
!= GST_FORMAT_BYTES
) {
316 FIXME("Not handling other format %i\n", seek_format
);
320 flush
= flags
& GST_SEEK_FLAG_FLUSH
;
321 seqnum
= gst_event_get_seqnum(event
);
323 /* send flush start */
325 tevent
= gst_event_new_flush_start();
326 gst_event_set_seqnum(tevent
, seqnum
);
327 gst_pad_push_event(This
->my_src
, tevent
);
328 if (This
->pInputPin
.pReader
)
329 IAsyncReader_BeginFlush(This
->pInputPin
.pReader
);
331 gst_pad_activate_push(This
->my_src
, 0);
334 This
->nextofs
= This
->start
= cur
;
336 /* and prepare to continue streaming */
338 tevent
= gst_event_new_flush_stop();
339 gst_event_set_seqnum(tevent
, seqnum
);
340 gst_pad_push_event(This
->my_src
, tevent
);
341 if (This
->pInputPin
.pReader
)
342 IAsyncReader_EndFlush(This
->pInputPin
.pReader
);
344 gst_pad_activate_push(This
->my_src
, 1);
350 static gboolean
event_src(GstPad
*pad
, GstEvent
*event
) {
351 GSTImpl
*This
= gst_pad_get_element_private(pad
);
352 switch (event
->type
) {
354 return gst_base_src_perform_seek(This
, event
);
355 case GST_EVENT_FLUSH_START
:
356 EnterCriticalSection(&This
->filter
.csFilter
);
357 if (This
->pInputPin
.pReader
)
358 IAsyncReader_BeginFlush(This
->pInputPin
.pReader
);
359 LeaveCriticalSection(&This
->filter
.csFilter
);
361 case GST_EVENT_FLUSH_STOP
:
362 EnterCriticalSection(&This
->filter
.csFilter
);
363 if (This
->pInputPin
.pReader
)
364 IAsyncReader_EndFlush(This
->pInputPin
.pReader
);
365 LeaveCriticalSection(&This
->filter
.csFilter
);
368 FIXME("%p (%u) stub\n", event
, event
->type
);
371 return gst_pad_event_default(pad
, event
);
376 static gboolean
event_sink(GstPad
*pad
, GstEvent
*event
) {
377 GSTOutPin
*pin
= gst_pad_get_element_private(pad
);
378 switch (event
->type
) {
379 case GST_EVENT_NEWSEGMENT
: {
381 gdouble rate
, applied_rate
;
383 gint64 start
, stop
, pos
;
384 gst_event_parse_new_segment_full(event
, &update
, &rate
, &applied_rate
, &format
, &start
, &stop
, &pos
);
385 if (format
!= GST_FORMAT_TIME
) {
386 FIXME("Ignoring new segment because of format %i\n", format
);
389 gst_segment_set_newsegment_full(pin
->segment
, update
, rate
, applied_rate
, format
, start
, stop
, pos
);
393 if (pin
->pin
.pin
.pConnectedTo
)
394 IPin_NewSegment(pin
->pin
.pin
.pConnectedTo
, pos
, stop
, rate
*applied_rate
);
398 if (pin
->pin
.pin
.pConnectedTo
)
399 IPin_EndOfStream(pin
->pin
.pin
.pConnectedTo
);
401 case GST_EVENT_FLUSH_START
:
402 if (pin
->pin
.pin
.pConnectedTo
)
403 IPin_BeginFlush(pin
->pin
.pin
.pConnectedTo
);
405 case GST_EVENT_FLUSH_STOP
:
406 gst_segment_init(pin
->segment
, GST_FORMAT_TIME
);
407 if (pin
->pin
.pin
.pConnectedTo
)
408 IPin_EndFlush(pin
->pin
.pin
.pConnectedTo
);
411 FIXME("%p stub %s\n", event
, gst_event_type_get_name(event
->type
));
412 return gst_pad_event_default(pad
, event
);
416 static void release_sample(void *data
) {
418 ret
= IMediaSample_Release((IMediaSample
*)data
);
419 TRACE("Releasing %p returns %u\n", data
, ret
);
422 static DWORD CALLBACK
push_data(LPVOID iface
) {
423 LONGLONG maxlen
, curlen
;
424 GSTImpl
*This
= iface
;
430 IAsyncReader_Length(This
->pInputPin
.pReader
, &maxlen
, &curlen
);
433 TRACE("Starting..\n");
435 REFERENCE_TIME tStart
, tStop
;
441 hr
= IMemAllocator_GetBuffer(This
->pInputPin
.pAlloc
, &buf
, NULL
, NULL
, 0);
445 if (This
->nextofs
>= maxlen
)
447 len
= IMediaSample_GetSize(buf
);
448 if (This
->nextofs
+ len
> maxlen
)
449 len
= maxlen
- This
->nextofs
;
451 tStart
= MEDIATIME_FROM_BYTES(This
->nextofs
);
452 tStop
= tStart
+ MEDIATIME_FROM_BYTES(len
);
453 IMediaSample_SetTime(buf
, &tStart
, &tStop
);
455 hr
= IAsyncReader_Request(This
->pInputPin
.pReader
, buf
, 0);
457 IMediaSample_Release(buf
);
460 This
->nextofs
+= len
;
461 hr
= IAsyncReader_WaitForNext(This
->pInputPin
.pReader
, -1, &buf
, &user
);
462 if (FAILED(hr
) || !buf
) {
464 IMediaSample_Release(buf
);
468 IMediaSample_GetPointer(buf
, &data
);
469 gstbuf
= gst_app_buffer_new(data
, IMediaSample_GetActualDataLength(buf
), release_sample
, buf
);
471 IMediaSample_Release(buf
);
474 gstbuf
->duration
= gstbuf
->timestamp
= -1;
475 ret
= gst_pad_push(This
->my_src
, gstbuf
);
479 ERR("Sending returned: %i\n", ret
);
480 if (ret
== GST_FLOW_ERROR
)
482 else if (ret
== GST_FLOW_WRONG_STATE
)
483 hr
= VFW_E_WRONG_STATE
;
484 else if (ret
== GST_FLOW_RESEND
)
490 gst_pad_push_event(This
->my_src
, gst_event_new_eos());
492 TRACE("Almost stopping.. %08x\n", hr
);
494 IAsyncReader_WaitForNext(This
->pInputPin
.pReader
, 0, &buf
, &user
);
496 IMediaSample_Release(buf
);
499 TRACE("Stopping.. %08x\n", hr
);
503 static HRESULT WINAPI
GST_OutPin_QueryAccept(IPin
*iface
, const AM_MEDIA_TYPE
*pmt
) {
504 GSTOutPin
*pin
= (GSTOutPin
*)iface
;
505 FIXME("stub %p\n", pin
);
509 static GstFlowReturn
got_data_sink(GstPad
*pad
, GstBuffer
*buf
) {
510 GSTOutPin
*pin
= gst_pad_get_element_private(pad
);
511 GSTImpl
*This
= (GSTImpl
*)pin
->pin
.pin
.pinInfo
.pFilter
;
512 IMediaSample
*sample
;
514 BOOL freeSamp
= FALSE
;
517 gst_buffer_unref(buf
);
518 TRACE("Triggering %p %p\n", pad
, pin
->caps_event
);
519 SetEvent(pin
->caps_event
);
520 return GST_FLOW_NOT_LINKED
;
523 if (GST_IS_APP_BUFFER(buf
)) {
524 sample
= GST_APP_BUFFER(buf
)->priv
;
525 TRACE("Pushing buffer\n");
526 } else if (buf
->parent
&& GST_IS_APP_BUFFER(buf
->parent
)) {
527 sample
= GST_APP_BUFFER(buf
->parent
)->priv
;
528 TRACE("Pushing sub-buffer\n");
531 hr
= BaseOutputPinImpl_GetDeliveryBuffer(&pin
->pin
, &sample
, NULL
, NULL
, 0);
533 if (hr
== VFW_E_NOT_CONNECTED
) {
534 gst_buffer_unref(buf
);
535 return GST_FLOW_NOT_LINKED
;
538 gst_buffer_unref(buf
);
539 ERR("Didn't get a GST_APP_BUFFER, and could not get a delivery buffer (%x), returning GST_FLOW_WRONG_STATE\n", hr
);
540 return GST_FLOW_WRONG_STATE
;
542 TRACE("Did not get a GST_APP_BUFFER, creating a sample\n");
543 IMediaSample_GetPointer(sample
, &ptr
);
544 memcpy(ptr
, GST_BUFFER_DATA(buf
), GST_BUFFER_SIZE(buf
));
546 IMediaSample_SetActualDataLength(sample
, GST_BUFFER_SIZE(buf
));
548 if (GST_BUFFER_TIMESTAMP_IS_VALID(buf
)) {
549 REFERENCE_TIME rtStart
= gst_segment_to_running_time(pin
->segment
, GST_FORMAT_TIME
, buf
->timestamp
);
553 if (GST_BUFFER_DURATION_IS_VALID(buf
)) {
554 REFERENCE_TIME tStart
= buf
->timestamp
/ 100;
555 REFERENCE_TIME tStop
= (buf
->timestamp
+ buf
->duration
) / 100;
556 REFERENCE_TIME rtStop
;
557 rtStop
= gst_segment_to_running_time(pin
->segment
, GST_FORMAT_TIME
, buf
->timestamp
+ buf
->duration
);
560 TRACE("Current time on %p: %i to %i ms\n", pin
, (int)(rtStart
/ 10000), (int)(rtStop
/ 10000));
561 IMediaSample_SetTime(sample
, &rtStart
, rtStop
>= 0 ? &rtStop
: NULL
);
562 IMediaSample_SetMediaTime(sample
, &tStart
, &tStop
);
564 IMediaSample_SetTime(sample
, rtStart
>= 0 ? &rtStart
: NULL
, NULL
);
565 IMediaSample_SetMediaTime(sample
, NULL
, NULL
);
568 IMediaSample_SetTime(sample
, NULL
, NULL
);
569 IMediaSample_SetMediaTime(sample
, NULL
, NULL
);
572 IMediaSample_SetDiscontinuity(sample
, GST_BUFFER_FLAG_IS_SET(buf
, GST_BUFFER_FLAG_DISCONT
));
573 IMediaSample_SetPreroll(sample
, GST_BUFFER_FLAG_IS_SET(buf
, GST_BUFFER_FLAG_PREROLL
));
574 IMediaSample_SetSyncPoint(sample
, !GST_BUFFER_FLAG_IS_SET(buf
, GST_BUFFER_FLAG_DELTA_UNIT
));
576 if (!pin
->pin
.pin
.pConnectedTo
)
577 hr
= VFW_E_NOT_CONNECTED
;
579 hr
= IMemInputPin_Receive(pin
->pin
.pMemInputPin
, sample
);
580 TRACE("sending sample: %08x\n", hr
);
581 gst_buffer_unref(buf
);
583 IMediaSample_Release(sample
);
584 if (hr
== VFW_E_NOT_CONNECTED
)
585 return GST_FLOW_NOT_LINKED
;
587 return GST_FLOW_WRONG_STATE
;
589 return GST_FLOW_RESEND
;
593 static GstFlowReturn
request_buffer_sink(GstPad
*pad
, guint64 ofs
, guint size
, GstCaps
*caps
, GstBuffer
**buf
) {
594 GSTOutPin
*pin
= gst_pad_get_element_private(pad
);
595 GSTImpl
*This
= (GSTImpl
*)pin
->pin
.pin
.pinInfo
.pFilter
;
596 IMediaSample
*sample
;
600 TRACE("Requesting buffer\n");
603 ret
= setcaps_sink(pad
, caps
);
605 return GST_FLOW_NOT_NEGOTIATED
;
606 *buf
= gst_buffer_new_and_alloc(size
);
610 if (caps
&& caps
!= GST_PAD_CAPS(pad
))
611 if (!setcaps_sink(pad
, caps
))
612 return GST_FLOW_NOT_NEGOTIATED
;
614 hr
= BaseOutputPinImpl_GetDeliveryBuffer(&pin
->pin
, &sample
, NULL
, NULL
, 0);
615 if (hr
== VFW_E_NOT_CONNECTED
)
616 return GST_FLOW_NOT_LINKED
;
618 ERR("Could not get output buffer: %08x\n", hr
);
620 return GST_FLOW_WRONG_STATE
;
622 IMediaSample_SetActualDataLength(sample
, size
);
623 IMediaSample_GetPointer(sample
, &ptr
);
624 *buf
= gst_app_buffer_new(ptr
, size
, release_sample
, sample
);
626 IMediaSample_Release(sample
);
627 ERR("Out of memory\n");
628 return GST_FLOW_ERROR
;
630 gst_buffer_set_caps(*buf
, caps
);
634 static GstFlowReturn
request_buffer_src(GstPad
*pad
, guint64 ofs
, guint len
, GstBuffer
**buf
) {
635 GSTImpl
*This
= gst_pad_get_element_private(pad
);
639 TRACE("Requesting %s %u\n", wine_dbgstr_longlong(ofs
), len
);
640 if (ofs
== (guint64
)-1)
641 ofs
= This
->nextpullofs
;
642 if (ofs
>= This
->filesize
) {
643 WARN("Reading past eof: %s, %u\n", wine_dbgstr_longlong(ofs
), len
);
644 return GST_FLOW_UNEXPECTED
;
646 if (len
+ ofs
> This
->filesize
)
647 len
= This
->filesize
- ofs
;
648 This
->nextpullofs
= ofs
+ len
;
650 ret
= gst_pad_alloc_buffer(This
->my_src
, ofs
, len
, NULL
, buf
);
653 hr
= IAsyncReader_SyncRead(This
->pInputPin
.pReader
, ofs
, len
, GST_BUFFER_DATA(*buf
));
655 ERR("Returned %08x\n", hr
);
656 return GST_FLOW_ERROR
;
662 static DWORD CALLBACK
push_data_init(LPVOID iface
) {
663 GSTImpl
*This
= iface
;
666 TRACE("Starting..\n");
669 GstFlowReturn ret
= request_buffer_src(This
->my_src
, ofs
, 4096, &buf
);
671 ERR("Obtaining buffer returned: %i\n", ret
);
674 ret
= gst_pad_push(This
->my_src
, buf
);
677 TRACE("Sending returned: %i\n", ret
);
681 TRACE("Stopping..\n");
685 static void removed_decoded_pad(GstElement
*bin
, GstPad
*pad
, GSTImpl
*This
) {
689 EnterCriticalSection(&This
->filter
.csFilter
);
690 for (x
= 0; x
< This
->cStreams
; ++x
) {
691 if (This
->ppPins
[x
]->their_src
== pad
)
694 if (x
== This
->cStreams
)
696 pin
= This
->ppPins
[x
];
697 gst_pad_unlink(pin
->their_src
, pin
->my_sink
);
698 gst_object_unref(pin
->their_src
);
699 pin
->their_src
= NULL
;
701 TRACE("Removed %i/%i\n", x
, This
->cStreams
);
702 LeaveCriticalSection(&This
->filter
.csFilter
);
705 static void init_new_decoded_pad(GstElement
*bin
, GstPad
*pad
, gboolean last
, GSTImpl
*This
) {
708 const char *typename
;
710 AM_MEDIA_TYPE amt
= {{0}};
716 int isvid
= 0, isaud
= 0;
718 piOutput
.dir
= PINDIR_OUTPUT
;
719 piOutput
.pFilter
= (IBaseFilter
*)This
;
720 name
= gst_pad_get_name(pad
);
721 MultiByteToWideChar(CP_UNIXCP
, 0, name
, -1, piOutput
.achName
, sizeof(piOutput
.achName
) / sizeof(piOutput
.achName
[0]) - 1);
722 TRACE("Name: %s\n", name
);
724 piOutput
.achName
[sizeof(piOutput
.achName
) / sizeof(piOutput
.achName
[0]) - 1] = 0;
726 caps
= gst_pad_get_caps_reffed(pad
);
727 arg
= gst_caps_get_structure(caps
, 0);
728 typename
= gst_structure_get_name(arg
);
730 mypad
= gst_pad_new(NULL
, GST_PAD_SINK
);
731 gst_pad_set_chain_function(mypad
, got_data_sink
);
732 gst_pad_set_event_function(mypad
, event_sink
);
733 gst_pad_set_bufferalloc_function(mypad
, request_buffer_sink
);
734 gst_pad_set_acceptcaps_function(mypad
, accept_caps_sink
);
735 gst_pad_set_setcaps_function(mypad
, setcaps_sink
);
737 if (!strcmp(typename
, "audio/x-raw-int") ||
738 !strcmp(typename
, "audio/x-raw-float")) {
740 } else if (!strcmp(typename
, "video/x-raw-rgb")
741 || !strcmp(typename
, "video/x-raw-yuv")) {
744 FIXME("Unknown type \'%s\'\n", typename
);
747 GST_PAD_CAPS(mypad
) = GST_CAPS_ANY
;
748 hr
= GST_AddPin(This
, &piOutput
, &amt
);
753 pin
= This
->ppPins
[This
->cStreams
- 1];
754 gst_pad_set_element_private(mypad
, pin
);
755 pin
->my_sink
= mypad
;
759 gst_segment_init(pin
->segment
, GST_FORMAT_TIME
);
760 ret
= gst_pad_link(pad
, mypad
);
761 gst_pad_activate_push(mypad
, 1);
762 TRACE("Linking: %i\n", ret
);
764 pin
->their_src
= pad
;
765 gst_object_ref(pin
->their_src
);
769 static void existing_new_pad(GstElement
*bin
, GstPad
*pad
, gboolean last
, GSTImpl
*This
) {
772 if (gst_pad_is_linked(pad
))
775 /* Still holding our own lock */
777 init_new_decoded_pad(bin
, pad
, last
, This
);
781 EnterCriticalSection(&This
->filter
.csFilter
);
782 for (x
= 0; x
< This
->cStreams
; ++x
) {
783 GSTOutPin
*pin
= This
->ppPins
[x
];
784 if (!pin
->their_src
) {
785 gst_segment_init(pin
->segment
, GST_FORMAT_TIME
);
786 if (gst_pad_link(pad
, pin
->my_sink
) >= 0) {
787 pin
->their_src
= pad
;
788 gst_object_ref(pin
->their_src
);
790 LeaveCriticalSection(&This
->filter
.csFilter
);
795 init_new_decoded_pad(bin
, pad
, last
, This
);
796 LeaveCriticalSection(&This
->filter
.csFilter
);
799 static gboolean
check_get_range(GstPad
*pad
) {
803 static gboolean
query_function(GstPad
*pad
, GstQuery
*query
) {
804 GSTImpl
*This
= gst_pad_get_element_private(pad
);
809 switch (GST_QUERY_TYPE(query
)) {
810 case GST_QUERY_DURATION
:
811 gst_query_parse_duration (query
, &format
, NULL
);
812 if (format
== GST_FORMAT_PERCENT
) {
813 gst_query_set_duration (query
, GST_FORMAT_PERCENT
, GST_FORMAT_PERCENT_MAX
);
816 ret
= gst_pad_query_convert (pad
, GST_FORMAT_BYTES
, This
->filesize
, &format
, &duration
);
817 gst_query_set_duration(query
, format
, duration
);
819 case GST_QUERY_SEEKING
:
820 gst_query_parse_seeking (query
, &format
, NULL
, NULL
, NULL
);
821 TRACE("Seeking %i %i\n", format
, GST_FORMAT_BYTES
);
822 if (format
!= GST_FORMAT_BYTES
)
824 gst_query_set_seeking(query
, GST_FORMAT_BYTES
, 1, 0, This
->filesize
);
827 FIXME("Unhandled query type %i\n", GST_QUERY_TYPE(query
));
829 case GST_QUERY_CONVERT
:
834 static gboolean
activate_push(GstPad
*pad
, gboolean activate
) {
835 GSTImpl
*This
= gst_pad_get_element_private(pad
);
836 EnterCriticalSection(&This
->filter
.csFilter
);
838 TRACE("Deactivating\n");
840 IAsyncReader_BeginFlush(This
->pInputPin
.pReader
);
841 if (This
->push_thread
) {
842 WaitForSingleObject(This
->push_thread
, -1);
843 CloseHandle(This
->push_thread
);
844 This
->push_thread
= NULL
;
847 IAsyncReader_EndFlush(This
->pInputPin
.pReader
);
848 if (This
->filter
.state
== State_Stopped
)
849 This
->nextofs
= This
->start
;
850 } else if (!This
->push_thread
) {
851 TRACE("Activating\n");
853 This
->push_thread
= CreateThread(NULL
, 0, push_data_init
, This
, 0, NULL
);
855 This
->push_thread
= CreateThread(NULL
, 0, push_data
, This
, 0, NULL
);
857 LeaveCriticalSection(&This
->filter
.csFilter
);
861 static void no_more_pads(GstElement
*decodebin
, GSTImpl
*This
) {
863 SetEvent(This
->event
);
867 GST_AUTOPLUG_SELECT_TRY
,
868 GST_AUTOPLUG_SELECT_EXPOSE
,
869 GST_AUTOPLUG_SELECT_SKIP
870 } GstAutoplugSelectResult
;
872 static GstAutoplugSelectResult
autoplug_blacklist(GstElement
*bin
, GstPad
*pad
, GstCaps
*caps
, GstElementFactory
*fact
, GSTImpl
*This
) {
873 const char *name
= gst_element_factory_get_longname(fact
);
875 if (strstr(name
, "Player protection")) {
876 WARN("Blacklisted a/52 decoder because it only works in Totem\n");
877 return GST_AUTOPLUG_SELECT_SKIP
;
879 if (!strcmp(name
, "Fluendo Hardware Accelerated Video Decoder")) {
880 WARN("Disabled video acceleration since it breaks in wine\n");
881 return GST_AUTOPLUG_SELECT_SKIP
;
883 TRACE("using \"%s\"\n", name
);
884 return GST_AUTOPLUG_SELECT_TRY
;
887 static GstBusSyncReply
watch_bus(GstBus
*bus
, GstMessage
*msg
, gpointer data
) {
888 GSTImpl
*This
= data
;
890 gchar
*dbg_info
= NULL
;
891 if (GST_MESSAGE_TYPE(msg
) & GST_MESSAGE_ERROR
) {
892 gst_message_parse_error(msg
, &err
, &dbg_info
);
893 FIXME("%s: %s\n", GST_OBJECT_NAME(msg
->src
), err
->message
);
894 WARN("%s\n", dbg_info
);
895 SetEvent(This
->event
);
896 } else if (GST_MESSAGE_TYPE(msg
) & GST_MESSAGE_WARNING
) {
897 gst_message_parse_warning(msg
, &err
, &dbg_info
);
898 WARN("%s: %s\n", GST_OBJECT_NAME(msg
->src
), err
->message
);
899 WARN("%s\n", dbg_info
);
907 static void unknown_type(GstElement
*bin
, GstPad
*pad
, GstCaps
*caps
, GSTImpl
*This
) {
908 gchar
*strcaps
= gst_caps_to_string(caps
);
909 FIXME("Could not find a filter for caps: %s\n", strcaps
);
913 static HRESULT
GST_Connect(GSTInPin
*pPin
, IPin
*pConnectPin
, ALLOCATOR_PROPERTIES
*props
) {
914 GSTImpl
*This
= (GSTImpl
*)pPin
->pin
.pinInfo
.pFilter
;
917 LONGLONG avail
, duration
;
918 GstFormat format
= GST_FORMAT_TIME
;
919 GstStaticPadTemplate src_template
= GST_STATIC_PAD_TEMPLATE(
923 GST_STATIC_CAPS_ANY
);
925 TRACE("%p %p %p\n", pPin
, pConnectPin
, props
);
926 This
->props
= *props
;
927 IAsyncReader_Length(pPin
->pReader
, &This
->filesize
, &avail
);
930 This
->bus
= gst_bus_new();
931 gst_bus_set_sync_handler(This
->bus
, watch_bus
, This
);
934 This
->gstfilter
= gst_element_factory_make("decodebin2", NULL
);
935 if (!This
->gstfilter
) {
936 FIXME("Could not make source filter, are gstreamer-plugins-* installed for %u bits?\n",
937 8 * (int)sizeof(void*));
940 gst_element_set_bus(This
->gstfilter
, This
->bus
);
941 g_signal_connect(This
->gstfilter
, "new-decoded-pad", G_CALLBACK(existing_new_pad
), This
);
942 g_signal_connect(This
->gstfilter
, "pad-removed", G_CALLBACK(removed_decoded_pad
), This
);
943 g_signal_connect(This
->gstfilter
, "autoplug-select", G_CALLBACK(autoplug_blacklist
), This
);
944 g_signal_connect(This
->gstfilter
, "unknown-type", G_CALLBACK(unknown_type
), This
);
946 This
->my_src
= gst_pad_new_from_static_template(&src_template
, "quartz-src");
947 gst_pad_set_getrange_function(This
->my_src
, request_buffer_src
);
948 gst_pad_set_checkgetrange_function(This
->my_src
, check_get_range
);
949 gst_pad_set_query_function(This
->my_src
, query_function
);
950 gst_pad_set_activatepush_function(This
->my_src
, activate_push
);
951 gst_pad_set_event_function(This
->my_src
, event_src
);
952 gst_pad_set_element_private (This
->my_src
, This
);
953 This
->their_sink
= gst_element_get_static_pad(This
->gstfilter
, "sink");
955 g_signal_connect(This
->gstfilter
, "no-more-pads", G_CALLBACK(no_more_pads
), This
);
956 ret
= gst_pad_link(This
->my_src
, This
->their_sink
);
957 gst_object_unref(This
->their_sink
);
959 ERR("Returns: %i\n", ret
);
962 This
->start
= This
->nextofs
= This
->nextpullofs
= This
->stop
= 0;
964 /* Add initial pins */
965 This
->initial
= This
->discont
= TRUE
;
966 ResetEvent(This
->event
);
967 gst_element_set_state(This
->gstfilter
, GST_STATE_PLAYING
);
968 gst_pad_set_active(This
->my_src
, 1);
969 WaitForSingleObject(This
->event
, -1);
970 gst_element_get_state(This
->gstfilter
, NULL
, NULL
, -1);
973 WARN("Ret: %i\n", ret
);
975 } else if (!This
->cStreams
) {
976 FIXME("GStreamer could not find any streams\n");
979 gst_pad_query_duration(This
->ppPins
[0]->their_src
, &format
, &duration
);
980 for (i
= 0; i
< This
->cStreams
; ++i
) {
981 This
->ppPins
[i
]->seek
.llDuration
= This
->ppPins
[i
]->seek
.llStop
= duration
/ 100;
982 This
->ppPins
[i
]->seek
.llCurrent
= 0;
983 if (!This
->ppPins
[i
]->seek
.llDuration
)
984 This
->ppPins
[i
]->seek
.dwCapabilities
= 0;
985 WaitForSingleObject(This
->ppPins
[i
]->caps_event
, -1);
989 *props
= This
->props
;
990 gst_element_set_state(This
->gstfilter
, GST_STATE_READY
);
991 gst_element_get_state(This
->gstfilter
, NULL
, NULL
, -1);
992 if (This
->push_thread
)
993 gst_pad_activate_push(This
->my_src
, 0);
995 This
->initial
= FALSE
;
996 This
->nextofs
= This
->nextpullofs
= 0;
1000 static inline GSTOutPin
*impl_from_IMediaSeeking( IMediaSeeking
*iface
) {
1001 return CONTAINING_RECORD(iface
, GSTOutPin
, seek
.IMediaSeeking_iface
);
1004 static IPin
* WINAPI
GST_GetPin(BaseFilter
*iface
, int pos
)
1006 GSTImpl
*This
= (GSTImpl
*)iface
;
1007 TRACE("Asking for pos %x\n", pos
);
1009 if (pos
> This
->cStreams
|| pos
< 0)
1013 IPin_AddRef((IPin
*)&This
->pInputPin
);
1014 return (IPin
*)&This
->pInputPin
;
1018 IPin_AddRef((IPin
*)This
->ppPins
[pos
- 1]);
1019 return (IPin
*)This
->ppPins
[pos
- 1];
1023 static LONG WINAPI
GST_GetPinCount(BaseFilter
*iface
)
1025 GSTImpl
*This
= (GSTImpl
*)iface
;
1026 return (This
->cStreams
+ 1);
1029 static const BaseFilterFuncTable BaseFuncTable
= {
1034 IUnknown
* CALLBACK
Gstreamer_Splitter_create(IUnknown
*punkout
, HRESULT
*phr
) {
1035 IUnknown
*obj
= NULL
;
1039 if (!Gstreamer_init())
1045 This
= CoTaskMemAlloc(sizeof(*This
));
1046 obj
= (IUnknown
*)This
;
1049 *phr
= E_OUTOFMEMORY
;
1053 BaseFilter_Init(&This
->filter
, &GST_Vtbl
, &CLSID_Gstreamer_Splitter
, (DWORD_PTR
)(__FILE__
": GSTImpl.csFilter"), &BaseFuncTable
);
1056 This
->ppPins
= NULL
;
1057 This
->push_thread
= NULL
;
1058 This
->event
= CreateEventW(NULL
, 0, 0, NULL
);
1061 piInput
= &This
->pInputPin
.pin
.pinInfo
;
1062 piInput
->dir
= PINDIR_INPUT
;
1063 piInput
->pFilter
= (IBaseFilter
*)This
;
1064 lstrcpynW(piInput
->achName
, wcsInputPinName
, sizeof(piInput
->achName
) / sizeof(piInput
->achName
[0]));
1065 This
->pInputPin
.pin
.IPin_iface
.lpVtbl
= &GST_InputPin_Vtbl
;
1066 This
->pInputPin
.pin
.refCount
= 1;
1067 This
->pInputPin
.pin
.pConnectedTo
= NULL
;
1068 This
->pInputPin
.pin
.pCritSec
= &This
->filter
.csFilter
;
1069 ZeroMemory(&This
->pInputPin
.pin
.mtCurrent
, sizeof(AM_MEDIA_TYPE
));
1074 static void GST_Destroy(GSTImpl
*This
) {
1075 IPin
*connected
= NULL
;
1079 TRACE("Destroying\n");
1081 CloseHandle(This
->event
);
1083 /* Don't need to clean up output pins, disconnecting input pin will do that */
1084 IPin_ConnectedTo((IPin
*)&This
->pInputPin
, &connected
);
1086 hr
= IPin_Disconnect(connected
);
1088 IPin_Release(connected
);
1089 hr
= IPin_Disconnect((IPin
*)&This
->pInputPin
);
1092 pinref
= IPin_Release((IPin
*)&This
->pInputPin
);
1094 /* Valgrind could find this, if I kill it here */
1095 ERR("pinref should be null, is %u, destroying anyway\n", pinref
);
1096 assert((LONG
)pinref
> 0);
1099 pinref
= IPin_Release((IPin
*)&This
->pInputPin
);
1102 gst_bus_set_sync_handler(This
->bus
, NULL
, NULL
);
1103 gst_object_unref(This
->bus
);
1105 BaseFilter_Destroy(&This
->filter
);
1106 CoTaskMemFree(This
);
1109 static HRESULT WINAPI
GST_QueryInterface(IBaseFilter
*iface
, REFIID riid
, LPVOID
*ppv
) {
1110 GSTImpl
*This
= (GSTImpl
*)iface
;
1111 TRACE("(%s, %p)\n", debugstr_guid(riid
), ppv
);
1115 if (IsEqualIID(riid
, &IID_IUnknown
))
1117 else if (IsEqualIID(riid
, &IID_IPersist
))
1119 else if (IsEqualIID(riid
, &IID_IMediaFilter
))
1121 else if (IsEqualIID(riid
, &IID_IBaseFilter
))
1125 IUnknown_AddRef((IUnknown
*)(*ppv
));
1129 if (!IsEqualIID(riid
, &IID_IPin
) && !IsEqualIID(riid
, &IID_IVideoWindow
) &&
1130 !IsEqualIID(riid
, &IID_IAMFilterMiscFlags
))
1131 FIXME("No interface for %s!\n", debugstr_guid(riid
));
1133 return E_NOINTERFACE
;
1136 static ULONG WINAPI
GST_Release(IBaseFilter
*iface
) {
1137 GSTImpl
*This
= (GSTImpl
*)iface
;
1138 ULONG refCount
= InterlockedDecrement(&This
->filter
.refCount
);
1140 TRACE("(%p)->() Release from %d\n", This
, refCount
+ 1);
1148 static HRESULT WINAPI
GST_Stop(IBaseFilter
*iface
) {
1149 GSTImpl
*This
= (GSTImpl
*)iface
;
1153 if (This
->gstfilter
)
1154 gst_element_set_state(This
->gstfilter
, GST_STATE_READY
);
1158 static HRESULT WINAPI
GST_Pause(IBaseFilter
*iface
) {
1160 GSTImpl
*This
= (GSTImpl
*)iface
;
1162 GstStateChangeReturn ret
;
1165 if (!This
->gstfilter
)
1166 return VFW_E_NOT_CONNECTED
;
1168 gst_element_get_state(This
->gstfilter
, &now
, NULL
, -1);
1169 if (now
== GST_STATE_PAUSED
)
1171 if (now
!= GST_STATE_PLAYING
)
1172 hr
= IBaseFilter_Run(iface
, -1);
1175 ret
= gst_element_set_state(This
->gstfilter
, GST_STATE_PAUSED
);
1176 if (ret
== GST_STATE_CHANGE_ASYNC
)
1181 static HRESULT WINAPI
GST_Run(IBaseFilter
*iface
, REFERENCE_TIME tStart
) {
1183 GSTImpl
*This
= (GSTImpl
*)iface
;
1186 HRESULT hr_any
= VFW_E_NOT_CONNECTED
;
1188 TRACE("(%s)\n", wine_dbgstr_longlong(tStart
));
1190 if (!This
->gstfilter
)
1191 return VFW_E_NOT_CONNECTED
;
1193 EnterCriticalSection(&This
->filter
.csFilter
);
1194 This
->filter
.rtStreamStart
= tStart
;
1195 LeaveCriticalSection(&This
->filter
.csFilter
);
1197 gst_element_get_state(This
->gstfilter
, &now
, NULL
, -1);
1198 if (now
== GST_STATE_PLAYING
)
1200 if (now
== GST_STATE_PAUSED
) {
1201 GstStateChangeReturn ret
;
1202 ret
= gst_element_set_state(This
->gstfilter
, GST_STATE_PLAYING
);
1203 if (ret
== GST_STATE_CHANGE_ASYNC
)
1208 EnterCriticalSection(&This
->filter
.csFilter
);
1209 gst_pad_set_blocked(This
->my_src
, 0);
1210 gst_pad_set_blocked(This
->their_sink
, 0);
1211 gst_element_set_state(This
->gstfilter
, GST_STATE_PLAYING
);
1212 This
->filter
.rtStreamStart
= tStart
;
1214 for (i
= 0; i
< This
->cStreams
; i
++) {
1215 hr
= BaseOutputPinImpl_Active((BaseOutputPin
*)This
->ppPins
[i
]);
1216 if (SUCCEEDED(hr
)) {
1217 gst_pad_set_blocked(This
->ppPins
[i
]->my_sink
, 0);
1218 if (This
->ppPins
[i
]->their_src
)
1219 gst_pad_set_blocked(This
->ppPins
[i
]->their_src
, 0);
1225 gst_pad_set_active(This
->my_src
, 1);
1226 LeaveCriticalSection(&This
->filter
.csFilter
);
1231 static HRESULT WINAPI
GST_GetState(IBaseFilter
*iface
, DWORD dwMilliSecsTimeout
, FILTER_STATE
*pState
) {
1232 GSTImpl
*This
= (GSTImpl
*)iface
;
1234 GstState now
, pending
;
1235 GstStateChangeReturn ret
;
1237 TRACE("(%d, %p)\n", dwMilliSecsTimeout
, pState
);
1239 if (!This
->gstfilter
) {
1240 *pState
= State_Stopped
;
1244 ret
= gst_element_get_state(This
->gstfilter
, &now
, &pending
, dwMilliSecsTimeout
== INFINITE
? -1 : dwMilliSecsTimeout
* 1000);
1246 if (ret
== GST_STATE_CHANGE_ASYNC
)
1247 hr
= VFW_S_STATE_INTERMEDIATE
;
1252 case GST_STATE_PAUSED
: *pState
= State_Paused
; return hr
;
1253 case GST_STATE_PLAYING
: *pState
= State_Running
; return hr
;
1254 default: *pState
= State_Stopped
; return hr
;
1258 static HRESULT WINAPI
GST_FindPin(IBaseFilter
*iface
, LPCWSTR Id
, IPin
**ppPin
) {
1259 FIXME("(%p)->(%s,%p) stub\n", iface
, debugstr_w(Id
), ppPin
);
1263 static const IBaseFilterVtbl GST_Vtbl
= {
1265 BaseFilterImpl_AddRef
,
1267 BaseFilterImpl_GetClassID
,
1272 BaseFilterImpl_SetSyncSource
,
1273 BaseFilterImpl_GetSyncSource
,
1274 BaseFilterImpl_EnumPins
,
1276 BaseFilterImpl_QueryFilterInfo
,
1277 BaseFilterImpl_JoinFilterGraph
,
1278 BaseFilterImpl_QueryVendorInfo
1281 static HRESULT WINAPI
GST_ChangeCurrent(IMediaSeeking
*iface
) {
1285 static HRESULT WINAPI
GST_ChangeStop(IMediaSeeking
*iface
) {
1289 static HRESULT WINAPI
GST_ChangeRate(IMediaSeeking
*iface
) {
1290 GSTOutPin
*This
= impl_from_IMediaSeeking(iface
);
1291 GstEvent
*ev
= gst_event_new_seek(This
->seek
.dRate
, GST_FORMAT_TIME
, 0, GST_SEEK_TYPE_NONE
, -1, GST_SEEK_TYPE_NONE
, -1);
1292 TRACE("(%p) New rate %g\n", iface
, This
->seek
.dRate
);
1293 gst_pad_push_event(This
->my_sink
, ev
);
1297 static HRESULT WINAPI
GST_Seeking_QueryInterface(IMediaSeeking
*iface
, REFIID riid
, void **ppv
) {
1298 GSTOutPin
*This
= impl_from_IMediaSeeking(iface
);
1299 return IUnknown_QueryInterface((IUnknown
*)This
, riid
, ppv
);
1302 static ULONG WINAPI
GST_Seeking_AddRef(IMediaSeeking
*iface
) {
1303 GSTOutPin
*This
= impl_from_IMediaSeeking(iface
);
1304 return IUnknown_AddRef((IUnknown
*)This
);
1307 static ULONG WINAPI
GST_Seeking_Release(IMediaSeeking
*iface
) {
1308 GSTOutPin
*This
= impl_from_IMediaSeeking(iface
);
1309 return IUnknown_Release((IUnknown
*)This
);
1312 static HRESULT WINAPI
GST_Seeking_GetCurrentPosition(IMediaSeeking
*iface
, REFERENCE_TIME
*pos
) {
1313 GSTOutPin
*This
= impl_from_IMediaSeeking(iface
);
1314 GstFormat format
= GST_FORMAT_TIME
;
1319 if (!This
->their_src
) {
1320 *pos
= This
->seek
.llCurrent
;
1321 TRACE("Cached value\n");
1322 if (This
->seek
.llDuration
)
1328 if (!gst_pad_query_position(This
->their_src
, &format
, pos
)) {
1329 WARN("Could not query position\n");
1333 This
->seek
.llCurrent
= *pos
;
1337 static GstSeekType
type_from_flags(DWORD flags
) {
1338 switch (flags
& AM_SEEKING_PositioningBitsMask
) {
1339 case AM_SEEKING_NoPositioning
: return GST_SEEK_TYPE_NONE
;
1340 case AM_SEEKING_AbsolutePositioning
: return GST_SEEK_TYPE_SET
;
1341 case AM_SEEKING_RelativePositioning
: return GST_SEEK_TYPE_CUR
;
1342 case AM_SEEKING_IncrementalPositioning
: return GST_SEEK_TYPE_END
;
1344 return GST_SEEK_TYPE_NONE
;
1348 static HRESULT WINAPI
GST_Seeking_SetPositions(IMediaSeeking
*iface
, REFERENCE_TIME
*pCur
, DWORD curflags
, REFERENCE_TIME
*pStop
, DWORD stopflags
) {
1350 GSTOutPin
*This
= impl_from_IMediaSeeking(iface
);
1352 GstSeekType curtype
, stoptype
;
1355 if (!This
->seek
.llDuration
)
1358 hr
= SourceSeekingImpl_SetPositions(iface
, pCur
, curflags
, pStop
, stopflags
);
1359 if (!This
->their_src
)
1362 curtype
= type_from_flags(curflags
);
1363 stoptype
= type_from_flags(stopflags
);
1364 if (curflags
& AM_SEEKING_SeekToKeyFrame
)
1365 f
|= GST_SEEK_FLAG_KEY_UNIT
;
1366 if (curflags
& AM_SEEKING_Segment
)
1367 f
|= GST_SEEK_FLAG_SEGMENT
;
1368 if (!(curflags
& AM_SEEKING_NoFlush
))
1369 f
|= GST_SEEK_FLAG_FLUSH
;
1371 e
= gst_event_new_seek(This
->seek
.dRate
, GST_FORMAT_TIME
, f
, curtype
, pCur
? *pCur
* 100 : -1, stoptype
, pStop
? *pStop
* 100 : -1);
1372 if (gst_pad_push_event(This
->my_sink
, e
))
1378 static const IMediaSeekingVtbl GST_Seeking_Vtbl
=
1380 GST_Seeking_QueryInterface
,
1382 GST_Seeking_Release
,
1383 SourceSeekingImpl_GetCapabilities
,
1384 SourceSeekingImpl_CheckCapabilities
,
1385 SourceSeekingImpl_IsFormatSupported
,
1386 SourceSeekingImpl_QueryPreferredFormat
,
1387 SourceSeekingImpl_GetTimeFormat
,
1388 SourceSeekingImpl_IsUsingTimeFormat
,
1389 SourceSeekingImpl_SetTimeFormat
,
1390 SourceSeekingImpl_GetDuration
,
1391 SourceSeekingImpl_GetStopPosition
,
1392 GST_Seeking_GetCurrentPosition
,
1393 SourceSeekingImpl_ConvertTimeFormat
,
1394 GST_Seeking_SetPositions
,
1395 SourceSeekingImpl_GetPositions
,
1396 SourceSeekingImpl_GetAvailable
,
1397 SourceSeekingImpl_SetRate
,
1398 SourceSeekingImpl_GetRate
,
1399 SourceSeekingImpl_GetPreroll
1402 static inline GSTOutPin
*impl_from_IQualityControl( IQualityControl
*iface
)
1404 return (GSTOutPin
*)CONTAINING_RECORD(iface
, GSTOutPin
, IQualityControl_iface
);
1407 static HRESULT WINAPI
GST_QualityControl_QueryInterface(IQualityControl
*iface
, REFIID riid
, void **ppv
)
1409 GSTOutPin
*pin
= impl_from_IQualityControl(iface
);
1410 return IPin_QueryInterface((IPin
*)pin
, riid
, ppv
);
1413 static ULONG WINAPI
GST_QualityControl_AddRef(IQualityControl
*iface
)
1415 GSTOutPin
*pin
= impl_from_IQualityControl(iface
);
1416 return IPin_AddRef((IPin
*)pin
);
1419 static ULONG WINAPI
GST_QualityControl_Release(IQualityControl
*iface
)
1421 GSTOutPin
*pin
= impl_from_IQualityControl(iface
);
1422 return IPin_Release((IPin
*)pin
);
1425 static HRESULT WINAPI
GST_QualityControl_Notify(IQualityControl
*iface
, IBaseFilter
*sender
, Quality qm
) {
1426 GSTOutPin
*pin
= impl_from_IQualityControl(iface
);
1427 REFERENCE_TIME late
= qm
.Late
;
1428 if (qm
.Late
< 0 && -qm
.Late
> qm
.TimeStamp
)
1429 late
= -qm
.TimeStamp
;
1430 gst_pad_push_event(pin
->my_sink
, gst_event_new_qos(1000./qm
.Proportion
, late
*100, qm
.TimeStamp
*100));
1434 static HRESULT WINAPI
GST_QualityControl_SetSink(IQualityControl
*iface
, IQualityControl
*tonotify
)
1440 static const IQualityControlVtbl GSTOutPin_QualityControl_Vtbl
= {
1441 GST_QualityControl_QueryInterface
,
1442 GST_QualityControl_AddRef
,
1443 GST_QualityControl_Release
,
1444 GST_QualityControl_Notify
,
1445 GST_QualityControl_SetSink
1448 static HRESULT WINAPI
GSTOutPin_QueryInterface(IPin
*iface
, REFIID riid
, void **ppv
) {
1449 GSTOutPin
*This
= (GSTOutPin
*)iface
;
1451 TRACE("(%s, %p)\n", debugstr_guid(riid
), ppv
);
1455 if (IsEqualIID(riid
, &IID_IUnknown
))
1457 else if (IsEqualIID(riid
, &IID_IPin
))
1459 else if (IsEqualIID(riid
, &IID_IMediaSeeking
))
1461 else if (IsEqualIID(riid
, &IID_IQualityControl
))
1462 *ppv
= &This
->IQualityControl_iface
;
1465 IUnknown_AddRef((IUnknown
*)(*ppv
));
1468 FIXME("No interface for %s!\n", debugstr_guid(riid
));
1469 return E_NOINTERFACE
;
1472 static ULONG WINAPI
GSTOutPin_Release(IPin
*iface
) {
1473 GSTOutPin
*This
= (GSTOutPin
*)iface
;
1474 ULONG refCount
= InterlockedDecrement(&This
->pin
.pin
.refCount
);
1475 TRACE("(%p)->() Release from %d\n", iface
, refCount
+ 1);
1478 if (This
->their_src
)
1479 gst_pad_unlink(This
->their_src
, This
->my_sink
);
1480 gst_object_unref(This
->my_sink
);
1481 CloseHandle(This
->caps_event
);
1482 DeleteMediaType(This
->pmt
);
1483 FreeMediaType(&This
->pin
.pin
.mtCurrent
);
1484 gst_segment_free(This
->segment
);
1485 if (This
->pin
.pAllocator
)
1486 IMemAllocator_Release(This
->pin
.pAllocator
);
1487 CoTaskMemFree(This
);
1493 static HRESULT WINAPI
GSTOutPin_GetMediaType(BasePin
*iface
, int iPosition
, AM_MEDIA_TYPE
*pmt
)
1495 GSTOutPin
*This
= (GSTOutPin
*)iface
;
1498 return E_INVALIDARG
;
1500 return VFW_S_NO_MORE_ITEMS
;
1501 CopyMediaType(pmt
, This
->pmt
);
1505 static HRESULT WINAPI
GSTOutPin_DecideBufferSize(BaseOutputPin
*iface
, IMemAllocator
*pAlloc
, ALLOCATOR_PROPERTIES
*ppropInputRequest
)
1511 static HRESULT WINAPI
GSTOutPin_DecideAllocator(BaseOutputPin
*iface
, IMemInputPin
*pPin
, IMemAllocator
**pAlloc
)
1514 GSTOutPin
*This
= (GSTOutPin
*)iface
;
1515 GSTImpl
*GSTfilter
= (GSTImpl
*)This
->pin
.pin
.pinInfo
.pFilter
;
1518 if (GSTfilter
->pInputPin
.pAlloc
)
1520 hr
= IMemInputPin_NotifyAllocator(pPin
, GSTfilter
->pInputPin
.pAlloc
, FALSE
);
1523 *pAlloc
= GSTfilter
->pInputPin
.pAlloc
;
1524 IMemAllocator_AddRef(*pAlloc
);
1528 hr
= VFW_E_NO_ALLOCATOR
;
1533 static HRESULT WINAPI
GSTOutPin_BreakConnect(BaseOutputPin
*This
)
1537 TRACE("(%p)->()\n", This
);
1539 EnterCriticalSection(This
->pin
.pCritSec
);
1540 if (!This
->pin
.pConnectedTo
|| !This
->pMemInputPin
)
1541 hr
= VFW_E_NOT_CONNECTED
;
1544 hr
= IPin_Disconnect(This
->pin
.pConnectedTo
);
1545 IPin_Disconnect((IPin
*)This
);
1547 LeaveCriticalSection(This
->pin
.pCritSec
);
1552 static const IPinVtbl GST_OutputPin_Vtbl
= {
1553 GSTOutPin_QueryInterface
,
1556 BaseOutputPinImpl_Connect
,
1557 BaseOutputPinImpl_ReceiveConnection
,
1558 BaseOutputPinImpl_Disconnect
,
1559 BasePinImpl_ConnectedTo
,
1560 BasePinImpl_ConnectionMediaType
,
1561 BasePinImpl_QueryPinInfo
,
1562 BasePinImpl_QueryDirection
,
1563 BasePinImpl_QueryId
,
1564 GST_OutPin_QueryAccept
,
1565 BasePinImpl_EnumMediaTypes
,
1566 BasePinImpl_QueryInternalConnections
,
1567 BaseOutputPinImpl_EndOfStream
,
1568 BaseOutputPinImpl_BeginFlush
,
1569 BaseOutputPinImpl_EndFlush
,
1570 BasePinImpl_NewSegment
1573 static const BaseOutputPinFuncTable output_BaseOutputFuncTable
= {
1576 BaseOutputPinImpl_AttemptConnection
,
1577 BasePinImpl_GetMediaTypeVersion
,
1578 GSTOutPin_GetMediaType
1580 GSTOutPin_DecideBufferSize
,
1581 GSTOutPin_DecideAllocator
,
1582 GSTOutPin_BreakConnect
1585 static HRESULT
GST_AddPin(GSTImpl
*This
, const PIN_INFO
*piOutput
, const AM_MEDIA_TYPE
*amt
) {
1587 This
->ppPins
= CoTaskMemRealloc(This
->ppPins
, (This
->cStreams
+ 1) * sizeof(IPin
*));
1589 hr
= BaseOutputPin_Construct(&GST_OutputPin_Vtbl
, sizeof(GSTOutPin
), piOutput
, &output_BaseOutputFuncTable
, &This
->filter
.csFilter
, (IPin
**)(This
->ppPins
+ This
->cStreams
));
1590 if (SUCCEEDED(hr
)) {
1591 GSTOutPin
*pin
= This
->ppPins
[This
->cStreams
];
1592 pin
->pmt
= CoTaskMemAlloc(sizeof(AM_MEDIA_TYPE
));
1593 CopyMediaType(pin
->pmt
, amt
);
1594 pin
->pin
.pin
.pinInfo
.pFilter
= (LPVOID
)This
;
1595 pin
->caps_event
= CreateEventW(NULL
, 0, 0, NULL
);
1596 pin
->segment
= gst_segment_new();
1598 pin
->IQualityControl_iface
.lpVtbl
= &GSTOutPin_QualityControl_Vtbl
;
1599 SourceSeeking_Init(&pin
->seek
, &GST_Seeking_Vtbl
, GST_ChangeStop
, GST_ChangeCurrent
, GST_ChangeRate
, &This
->filter
.csFilter
);
1600 BaseFilterImpl_IncrementPinVersion((BaseFilter
*)This
);
1602 ERR("Failed with error %x\n", hr
);
1606 static HRESULT
GST_RemoveOutputPins(GSTImpl
*This
) {
1609 GSTOutPin
**ppOldPins
= This
->ppPins
;
1610 TRACE("(%p)\n", This
);
1612 if (!This
->gstfilter
)
1614 gst_element_set_bus(This
->gstfilter
, NULL
);
1615 gst_element_set_state(This
->gstfilter
, GST_STATE_NULL
);
1616 gst_pad_unlink(This
->my_src
, This
->their_sink
);
1617 if (This
->push_thread
)
1618 gst_pad_activate_push(This
->my_src
, 0);
1619 gst_object_unref(This
->my_src
);
1620 This
->my_src
= This
->their_sink
= NULL
;
1622 for (i
= 0; i
< This
->cStreams
; i
++) {
1623 hr
= BaseOutputPinImpl_BreakConnect(&ppOldPins
[i
]->pin
);
1624 TRACE("Disconnect: %08x\n", hr
);
1625 IPin_Release((IPin
*)ppOldPins
[i
]);
1628 This
->ppPins
= NULL
;
1629 gst_object_unref(This
->gstfilter
);
1630 This
->gstfilter
= NULL
;
1631 BaseFilterImpl_IncrementPinVersion((BaseFilter
*)This
);
1632 CoTaskMemFree(ppOldPins
);
1636 static ULONG WINAPI
GSTInPin_Release(IPin
*iface
) {
1637 GSTInPin
*This
= (GSTInPin
*)iface
;
1638 ULONG refCount
= InterlockedDecrement(&This
->pin
.refCount
);
1640 TRACE("(%p)->() Release from %d\n", iface
, refCount
+ 1);
1642 FreeMediaType(&This
->pin
.mtCurrent
);
1644 IMemAllocator_Release(This
->pAlloc
);
1645 This
->pAlloc
= NULL
;
1646 This
->pin
.IPin_iface
.lpVtbl
= NULL
;
1652 static HRESULT WINAPI
GSTInPin_ReceiveConnection(IPin
*iface
, IPin
*pReceivePin
, const AM_MEDIA_TYPE
*pmt
) {
1653 PIN_DIRECTION pindirReceive
;
1655 GSTInPin
*This
= (GSTInPin
*)iface
;
1657 TRACE("(%p/%p)->(%p, %p)\n", This
, iface
, pReceivePin
, pmt
);
1658 dump_AM_MEDIA_TYPE(pmt
);
1660 EnterCriticalSection(This
->pin
.pCritSec
);
1661 if (!This
->pin
.pConnectedTo
) {
1662 ALLOCATOR_PROPERTIES props
;
1665 props
.cbBuffer
= 16384;
1669 if (SUCCEEDED(hr
) && IPin_QueryAccept(iface
, pmt
) != S_OK
)
1670 hr
= VFW_E_TYPE_NOT_ACCEPTED
;
1671 if (SUCCEEDED(hr
)) {
1672 IPin_QueryDirection(pReceivePin
, &pindirReceive
);
1673 if (pindirReceive
!= PINDIR_OUTPUT
) {
1674 ERR("Can't connect from non-output pin\n");
1675 hr
= VFW_E_INVALID_DIRECTION
;
1679 This
->pReader
= NULL
;
1680 This
->pAlloc
= NULL
;
1682 hr
= IPin_QueryInterface(pReceivePin
, &IID_IAsyncReader
, (LPVOID
*)&This
->pReader
);
1684 hr
= GST_Connect(This
, pReceivePin
, &props
);
1686 hr
= IAsyncReader_RequestAllocator(This
->pReader
, NULL
, &props
, &This
->pAlloc
);
1687 if (SUCCEEDED(hr
)) {
1688 CopyMediaType(&This
->pin
.mtCurrent
, pmt
);
1689 This
->pin
.pConnectedTo
= pReceivePin
;
1690 IPin_AddRef(pReceivePin
);
1691 hr
= IMemAllocator_Commit(This
->pAlloc
);
1693 GST_RemoveOutputPins((GSTImpl
*)This
->pin
.pinInfo
.pFilter
);
1695 IAsyncReader_Release(This
->pReader
);
1696 This
->pReader
= NULL
;
1698 IMemAllocator_Release(This
->pAlloc
);
1699 This
->pAlloc
= NULL
;
1701 TRACE("Size: %i\n", props
.cbBuffer
);
1703 hr
= VFW_E_ALREADY_CONNECTED
;
1704 LeaveCriticalSection(This
->pin
.pCritSec
);
1708 static HRESULT WINAPI
GSTInPin_Disconnect(IPin
*iface
) {
1710 GSTInPin
*This
= (GSTInPin
*)iface
;
1714 hr
= IBaseFilter_GetState(This
->pin
.pinInfo
.pFilter
, INFINITE
, &state
);
1715 EnterCriticalSection(This
->pin
.pCritSec
);
1716 if (This
->pin
.pConnectedTo
) {
1717 GSTImpl
*Parser
= (GSTImpl
*)This
->pin
.pinInfo
.pFilter
;
1719 if (SUCCEEDED(hr
) && state
== State_Stopped
) {
1720 IMemAllocator_Decommit(This
->pAlloc
);
1721 IPin_Disconnect(This
->pin
.pConnectedTo
);
1722 This
->pin
.pConnectedTo
= NULL
;
1723 hr
= GST_RemoveOutputPins(Parser
);
1725 hr
= VFW_E_NOT_STOPPED
;
1728 LeaveCriticalSection(This
->pin
.pCritSec
);
1732 static HRESULT WINAPI
GSTInPin_QueryAccept(IPin
*iface
, const AM_MEDIA_TYPE
*pmt
) {
1733 GSTInPin
*This
= (GSTInPin
*)iface
;
1735 TRACE("(%p)->(%p)\n", This
, pmt
);
1736 dump_AM_MEDIA_TYPE(pmt
);
1738 if (IsEqualIID(&pmt
->majortype
, &MEDIATYPE_Stream
))
1743 static HRESULT WINAPI
GSTInPin_EndOfStream(IPin
*iface
) {
1744 GSTInPin
*pin
= (GSTInPin
*)iface
;
1745 GSTImpl
*This
= (GSTImpl
*)pin
->pin
.pinInfo
.pFilter
;
1747 FIXME("Propagate message on %p\n", This
);
1751 static HRESULT WINAPI
GSTInPin_BeginFlush(IPin
*iface
) {
1752 GSTInPin
*pin
= (GSTInPin
*)iface
;
1753 GSTImpl
*This
= (GSTImpl
*)pin
->pin
.pinInfo
.pFilter
;
1755 FIXME("Propagate message on %p\n", This
);
1759 static HRESULT WINAPI
GSTInPin_EndFlush(IPin
*iface
) {
1760 GSTInPin
*pin
= (GSTInPin
*)iface
;
1761 GSTImpl
*This
= (GSTImpl
*)pin
->pin
.pinInfo
.pFilter
;
1763 FIXME("Propagate message on %p\n", This
);
1767 static HRESULT WINAPI
GSTInPin_NewSegment(IPin
*iface
, REFERENCE_TIME tStart
, REFERENCE_TIME tStop
, double dRate
) {
1768 GSTInPin
*pin
= (GSTInPin
*)iface
;
1769 GSTImpl
*This
= (GSTImpl
*)pin
->pin
.pinInfo
.pFilter
;
1771 BasePinImpl_NewSegment(iface
, tStart
, tStop
, dRate
);
1772 FIXME("Propagate message on %p\n", This
);
1776 static HRESULT WINAPI
GSTInPin_QueryInterface(IPin
* iface
, REFIID riid
, LPVOID
* ppv
)
1778 GSTInPin
*This
= (GSTInPin
*)iface
;
1780 TRACE("(%p/%p)->(%s, %p)\n", This
, iface
, debugstr_guid(riid
), ppv
);
1784 if (IsEqualIID(riid
, &IID_IUnknown
))
1786 else if (IsEqualIID(riid
, &IID_IPin
))
1788 else if (IsEqualIID(riid
, &IID_IMediaSeeking
))
1790 return IBaseFilter_QueryInterface(This
->pin
.pinInfo
.pFilter
, &IID_IMediaSeeking
, ppv
);
1795 IUnknown_AddRef((IUnknown
*)(*ppv
));
1799 FIXME("No interface for %s!\n", debugstr_guid(riid
));
1801 return E_NOINTERFACE
;
1804 static HRESULT WINAPI
GSTInPin_EnumMediaTypes(IPin
*iface
, IEnumMediaTypes
**ppEnum
)
1806 BasePin
*This
= (BasePin
*)iface
;
1808 TRACE("(%p/%p)->(%p)\n", This
, iface
, ppEnum
);
1810 return EnumMediaTypes_Construct(This
, BasePinImpl_GetMediaType
, BasePinImpl_GetMediaTypeVersion
, ppEnum
);
1813 static const IPinVtbl GST_InputPin_Vtbl
= {
1814 GSTInPin_QueryInterface
,
1817 BaseInputPinImpl_Connect
,
1818 GSTInPin_ReceiveConnection
,
1819 GSTInPin_Disconnect
,
1820 BasePinImpl_ConnectedTo
,
1821 BasePinImpl_ConnectionMediaType
,
1822 BasePinImpl_QueryPinInfo
,
1823 BasePinImpl_QueryDirection
,
1824 BasePinImpl_QueryId
,
1825 GSTInPin_QueryAccept
,
1826 GSTInPin_EnumMediaTypes
,
1827 BasePinImpl_QueryInternalConnections
,
1828 GSTInPin_EndOfStream
,
1829 GSTInPin_BeginFlush
,