summaryrefslogtreecommitdiffstats
path: root/ext/soundtouch/gstpitch.cc
diff options
context:
space:
mode:
Diffstat (limited to 'ext/soundtouch/gstpitch.cc')
-rw-r--r--ext/soundtouch/gstpitch.cc107
1 files changed, 105 insertions, 2 deletions
diff --git a/ext/soundtouch/gstpitch.cc b/ext/soundtouch/gstpitch.cc
index b1a019f2..ad53f477 100644
--- a/ext/soundtouch/gstpitch.cc
+++ b/ext/soundtouch/gstpitch.cc
@@ -44,6 +44,8 @@ struct _GstPitchPrivate
{
gfloat stream_time_ratio;
+ GstEvent *pending_segment;
+
soundtouch::SoundTouch * st;
};
@@ -108,7 +110,7 @@ gst_pitch_base_init (gpointer g_class)
gst_element_class_set_details_simple (gstelement_class, "Pitch controller",
"Filter/Converter/Audio", "Control the pitch of an audio stream",
- "Wouter Paesen <wouter@kangaroot.net>");
+ "Wouter Paesen <wouter@blue-gate.be>");
}
static void
@@ -448,6 +450,11 @@ gst_pitch_convert (GstPitch * pitch,
return FALSE;
}
+ if (src_format == *dst_format) {
+ *dst_value = src_value;
+ return TRUE;
+ }
+
switch (src_format) {
case GST_FORMAT_BYTES:
switch (*dst_format) {
@@ -599,6 +606,67 @@ gst_pitch_src_query (GstPad * pad, GstQuery * query)
return res;
}
+/* this function returns FALSE if not enough data is known to transform the
+ * segment into proper downstream values. If the function does return false
+ * the sgement should be stalled until enough information is available.
+ * If the funtion returns TRUE, event will be replaced by the new downstream
+ * compatible event.
+ */
+static gboolean
+gst_pitch_process_segment (GstPitch * pitch, GstEvent ** event)
+{
+ GstFormat format, conv_format;
+ gint64 start_value, stop_value, base;
+ gint64 next_offset = 0, next_time = 0;
+ gboolean update = FALSE;
+ gdouble rate;
+ gfloat stream_time_ratio;
+
+ g_return_val_if_fail (event, FALSE);
+
+ GST_OBJECT_LOCK (pitch);
+ stream_time_ratio = pitch->priv->stream_time_ratio;
+ GST_OBJECT_UNLOCK (pitch);
+
+ gst_event_parse_new_segment (*event, &update, &rate, &format, &start_value,
+ &stop_value, &base);
+
+ GST_LOG_OBJECT (pitch->sinkpad, "segment %lld - %lld (%d)", start_value,
+ stop_value, format);
+
+ if (stream_time_ratio == 0) {
+ GST_LOG_OBJECT (pitch->sinkpad, "stream_time_ratio is zero");
+ return FALSE;
+ }
+
+ start_value = (gint64) (start_value / stream_time_ratio);
+ stop_value = (gint64) (stop_value / stream_time_ratio);
+ base = (gint64) (base / stream_time_ratio);
+
+ conv_format = GST_FORMAT_TIME;
+ if (!gst_pitch_convert (pitch, format, start_value, &conv_format, &next_time)) {
+ GST_LOG_OBJECT (pitch->sinkpad,
+ "could not convert segment start value to time");
+ return FALSE;
+ }
+
+ conv_format = GST_FORMAT_DEFAULT;
+ if (!gst_pitch_convert (pitch, format, start_value, &conv_format,
+ &next_offset)) {
+ GST_LOG_OBJECT (pitch->sinkpad,
+ "could not convert segment start value to offset");
+ return FALSE;
+ }
+
+ pitch->next_buffer_time = next_time;
+ pitch->next_buffer_offset = next_offset;
+
+ gst_event_unref (*event);
+ *event = gst_event_new_new_segment (update, rate, format, start_value,
+ stop_value, base);
+
+ return TRUE;
+}
static gboolean
gst_pitch_sink_event (GstPad * pad, GstEvent * event)
@@ -617,12 +685,22 @@ gst_pitch_sink_event (GstPad * pad, GstEvent * event)
case GST_EVENT_EOS:
gst_pitch_flush_buffer (pitch, TRUE);
break;
+ case GST_EVENT_NEWSEGMENT:
+ if (!gst_pitch_process_segment (pitch, &event)) {
+ GST_LOG_OBJECT (pad, "not enough data known, stalling segment");
+ if (GST_PITCH_GET_PRIVATE (pitch)->pending_segment)
+ gst_event_unref (GST_PITCH_GET_PRIVATE (pitch)->pending_segment);
+ GST_PITCH_GET_PRIVATE (pitch)->pending_segment = event;
+ event = NULL;
+ }
+ break;
default:
break;
}
/* and forward it */
- res = gst_pad_event_default (pad, event);
+ if (event)
+ res = gst_pad_event_default (pad, event);
gst_object_unref (pitch);
return res;
@@ -643,6 +721,25 @@ gst_pitch_chain (GstPad * pad, GstBuffer * buffer)
GST_LOG_OBJECT (pitch, "incoming buffer (%d samples)",
(gint) (GST_BUFFER_SIZE (buffer) / pitch->sample_size));
+ if (GST_PITCH_GET_PRIVATE (pitch)->pending_segment) {
+ GstEvent *event =
+ gst_event_copy (GST_PITCH_GET_PRIVATE (pitch)->pending_segment);
+
+ GST_LOG_OBJECT (pitch, "processing stalled segment");
+ if (!gst_pitch_process_segment (pitch, &event)) {
+ gst_event_unref (event);
+ return GST_FLOW_ERROR;
+ }
+
+ if (!gst_pad_event_default (pitch->sinkpad, event)) {
+ gst_event_unref (event);
+ return GST_FLOW_ERROR;
+ }
+
+ gst_event_unref (GST_PITCH_GET_PRIVATE (pitch)->pending_segment);
+ GST_PITCH_GET_PRIVATE (pitch)->pending_segment = NULL;
+ }
+
priv->st->putSamples ((gfloat *) GST_BUFFER_DATA (buffer),
GST_BUFFER_SIZE (buffer) / pitch->sample_size);
gst_buffer_unref (buffer);
@@ -683,7 +780,13 @@ gst_pitch_change_state (GstElement * element, GstStateChange transition)
switch (transition) {
case GST_STATE_CHANGE_PLAYING_TO_PAUSED:
+ break;
case GST_STATE_CHANGE_PAUSED_TO_READY:
+ if (GST_PITCH_GET_PRIVATE (pitch)->pending_segment) {
+ gst_event_unref (GST_PITCH_GET_PRIVATE (pitch)->pending_segment);
+ GST_PITCH_GET_PRIVATE (pitch)->pending_segment = NULL;
+ }
+ break;
case GST_STATE_CHANGE_READY_TO_NULL:
default:
break;