summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorDavid Schleef <ds@schleef.org>2012-07-02 17:55:37 -0700
committerDavid Schleef <ds@schleef.org>2012-07-02 18:59:34 -0700
commit3f799ce916f18b02efed62e28aef3982c8de5f5d (patch)
treef142a25ba9d4007b0fa2e402a230dd06936136d8
parentde71b467f2927b3bc7ce7c996d76ee58b41f498e (diff)
decklinksink: Better handling of audio output
-rw-r--r--sys/decklink/gstdecklinksink.cpp81
-rw-r--r--sys/decklink/gstdecklinksink.h9
2 files changed, 65 insertions, 25 deletions
diff --git a/sys/decklink/gstdecklinksink.cpp b/sys/decklink/gstdecklinksink.cpp
index 458b1b9c7..e1933f323 100644
--- a/sys/decklink/gstdecklinksink.cpp
+++ b/sys/decklink/gstdecklinksink.cpp
@@ -299,6 +299,7 @@ gst_decklink_sink_init (GstDecklinkSink * decklinksink,
decklinksink->cond = g_cond_new ();
decklinksink->mutex = g_mutex_new ();
decklinksink->audio_mutex = g_mutex_new ();
+ decklinksink->audio_cond = g_cond_new ();
decklinksink->mode = GST_DECKLINK_MODE_NTSC;
decklinksink->device = 0;
@@ -430,6 +431,7 @@ gst_decklink_sink_finalize (GObject * object)
/* clean up object here */
g_cond_free (decklinksink->cond);
g_mutex_free (decklinksink->mutex);
+ g_cond_free (decklinksink->audio_cond);
g_mutex_free (decklinksink->audio_mutex);
delete decklinksink->callback;
@@ -462,14 +464,15 @@ gst_decklink_sink_start (GstDecklinkSink * decklinksink)
BMDAudioSampleType sample_depth;
decklinksink->decklink = gst_decklink_get_nth_device (decklinksink->device);
- if (decklinksink->decklink) {
+ if (!decklinksink->decklink) {
+ GST_WARNING ("failed to get device %d", decklinksink->device);
return FALSE;
}
ret = decklinksink->decklink->QueryInterface (IID_IDeckLinkOutput,
(void **) &decklinksink->output);
if (ret != S_OK) {
- GST_ERROR ("selected device does not have output interface");
+ GST_WARNING ("selected device does not have output interface");
return FALSE;
}
@@ -480,7 +483,7 @@ gst_decklink_sink_start (GstDecklinkSink * decklinksink)
ret = decklinksink->output->EnableVideoOutput (mode->mode,
bmdVideoOutputFlagDefault);
if (ret != S_OK) {
- GST_ERROR ("failed to enable video output");
+ GST_WARNING ("failed to enable video output");
return FALSE;
}
//decklinksink->video_enabled = TRUE;
@@ -492,10 +495,10 @@ gst_decklink_sink_start (GstDecklinkSink * decklinksink)
ret = decklinksink->output->EnableAudioOutput (bmdAudioSampleRate48kHz,
sample_depth, 2, bmdAudioOutputStreamContinuous);
if (ret != S_OK) {
- GST_ERROR ("failed to enable audio output");
+ GST_WARNING ("failed to enable audio output");
return FALSE;
}
- decklinksink->audio_buffer = gst_buffer_new ();
+ decklinksink->audio_adapter = gst_adapter_new ();
decklinksink->num_frames = 0;
@@ -510,6 +513,10 @@ gst_decklink_sink_force_stop (GstDecklinkSink * decklinksink)
g_cond_signal (decklinksink->cond);
g_mutex_unlock (decklinksink->mutex);
+ g_mutex_lock (decklinksink->audio_mutex);
+ g_cond_signal (decklinksink->audio_cond);
+ g_mutex_unlock (decklinksink->audio_mutex);
+
return TRUE;
}
@@ -748,7 +755,7 @@ gst_decklink_sink_videosink_chain (GstPad * pad, GstBuffer * buffer)
ret = decklinksink->output->EnableVideoOutput (decklinksink->display_mode,
bmdVideoOutputFlagDefault);
if (ret != S_OK) {
- GST_ERROR ("failed to enable video output");
+ GST_WARNING ("failed to enable video output");
//return FALSE;
}
decklinksink->video_enabled = TRUE;
@@ -820,6 +827,18 @@ gst_decklink_sink_videosink_event (GstPad * pad, GstEvent * event)
GST_DEBUG_OBJECT (decklinksink, "event");
switch (GST_EVENT_TYPE (event)) {
+ case GST_EVENT_EOS:
+ decklinksink->video_eos = TRUE;
+ decklinksink->video_seqnum = gst_event_get_seqnum (event);
+ {
+ GstMessage *message;
+
+ message = gst_message_new_eos (GST_OBJECT_CAST (decklinksink));
+ gst_message_set_seqnum (message, decklinksink->video_seqnum);
+ gst_element_post_message (GST_ELEMENT_CAST (decklinksink), message);
+ }
+
+ break;
default:
res = gst_pad_event_default (pad, event);
break;
@@ -1001,14 +1020,18 @@ gst_decklink_sink_audiosink_chain (GstPad * pad, GstBuffer * buffer)
GST_DEBUG_OBJECT (decklinksink, "chain");
- // concatenate both buffers
+ if (decklinksink->stop) {
+ return GST_FLOW_WRONG_STATE;
+ }
+
g_mutex_lock (decklinksink->audio_mutex);
- decklinksink->audio_buffer =
- gst_buffer_join (decklinksink->audio_buffer, buffer);
+ while (!decklinksink->stop &&
+ gst_adapter_available (decklinksink->audio_adapter) > 1600 * 4 * 2) {
+ g_cond_wait (decklinksink->audio_cond, decklinksink->audio_mutex);
+ }
+ gst_adapter_push (decklinksink->audio_adapter, buffer);
g_mutex_unlock (decklinksink->audio_mutex);
- // GST_DEBUG("Audio Buffer Size: %d", GST_BUFFER_SIZE (decklinksink->audio_buffer));
-
gst_object_unref (decklinksink);
ret = GST_FLOW_OK;
@@ -1040,6 +1063,10 @@ gst_decklink_sink_audiosink_event (GstPad * pad, GstEvent * event)
GST_DEBUG_OBJECT (decklinksink, "event");
switch (GST_EVENT_TYPE (event)) {
+ case GST_EVENT_EOS:
+ decklinksink->audio_eos = TRUE;
+ decklinksink->audio_seqnum = gst_event_get_seqnum (event);
+ break;
default:
res = gst_pad_event_default (pad, event);
break;
@@ -1122,7 +1149,7 @@ HRESULT
HRESULT
Output::ScheduledPlaybackHasStopped ()
{
- GST_ERROR ("ScheduledPlaybackHasStopped");
+ GST_DEBUG ("ScheduledPlaybackHasStopped");
return S_OK;
}
@@ -1130,7 +1157,6 @@ HRESULT
Output::RenderAudioSamples (bool preroll)
{
uint32_t samplesWritten;
- GstBuffer *buffer;
// guint64 samplesToWrite;
@@ -1139,22 +1165,31 @@ Output::RenderAudioSamples (bool preroll)
decklinksink->output->BeginAudioPreroll ();
// running = true;
} else {
+ int n;
+ const guint8 *data;
+
g_mutex_lock (decklinksink->audio_mutex);
- decklinksink->output->ScheduleAudioSamples (GST_BUFFER_DATA (decklinksink->audio_buffer), GST_BUFFER_SIZE (decklinksink->audio_buffer) / 4, // 2 bytes per sample, stereo
- 0, 0, &samplesWritten);
- buffer =
- gst_buffer_new_and_alloc (GST_BUFFER_SIZE (decklinksink->audio_buffer) -
- (samplesWritten * 4));
+ n = gst_adapter_available (decklinksink->audio_adapter);
+ if (n > 0) {
+ data = gst_adapter_peek (decklinksink->audio_adapter, n);
- memcpy (GST_BUFFER_DATA (buffer),
- GST_BUFFER_DATA (decklinksink->audio_buffer) + (samplesWritten * 4),
- GST_BUFFER_SIZE (decklinksink->audio_buffer) - (samplesWritten * 4));
+ decklinksink->output->ScheduleAudioSamples ((void *) data, n / 4,
+ 0, 0, &samplesWritten);
- gst_buffer_unref (decklinksink->audio_buffer);
+ gst_adapter_flush (decklinksink->audio_adapter, samplesWritten * 4);
+ GST_DEBUG ("wrote %d samples, %d available", samplesWritten, n / 4);
- decklinksink->audio_buffer = buffer;
+ g_cond_signal (decklinksink->audio_cond);
+ } else {
+ if (decklinksink->audio_eos) {
+ GstMessage *message;
+ message = gst_message_new_eos (GST_OBJECT_CAST (decklinksink));
+ gst_message_set_seqnum (message, decklinksink->audio_seqnum);
+ gst_element_post_message (GST_ELEMENT_CAST (decklinksink), message);
+ }
+ }
g_mutex_unlock (decklinksink->audio_mutex);
}
diff --git a/sys/decklink/gstdecklinksink.h b/sys/decklink/gstdecklinksink.h
index 20c47edc2..8969303e7 100644
--- a/sys/decklink/gstdecklinksink.h
+++ b/sys/decklink/gstdecklinksink.h
@@ -21,6 +21,7 @@
#define _GST_DECKLINK_SINK_H_
#include <gst/gst.h>
+#include <gst/base/gstadapter.h>
#include "gstdecklink.h"
G_BEGIN_DECLS
@@ -51,7 +52,7 @@ public IDeckLinkAudioOutputCallback
struct _GstDecklinkSink
{
GstElement base_decklinksink;
- GstBuffer *audio_buffer;
+ GstAdapter *audio_adapter;
GstPad *videosinkpad;
GstPad *audiosinkpad;
@@ -59,9 +60,13 @@ struct _GstDecklinkSink
GMutex *mutex;
GCond *cond;
GMutex *audio_mutex;
-// GCond *audio_cond;
+ GCond *audio_cond;
int queued_frames;
gboolean stop;
+ gboolean video_eos;
+ gboolean audio_eos;
+ int video_seqnum;
+ int audio_seqnum;
IDeckLink *decklink;
IDeckLinkOutput *output;