summaryrefslogtreecommitdiffstats
path: root/ext
diff options
context:
space:
mode:
Diffstat (limited to 'ext')
-rw-r--r--ext/wavpack/gstwavpackparse.c237
-rw-r--r--ext/wavpack/gstwavpackparse.h3
2 files changed, 190 insertions, 50 deletions
diff --git a/ext/wavpack/gstwavpackparse.c b/ext/wavpack/gstwavpackparse.c
index b311af38..720b25b0 100644
--- a/ext/wavpack/gstwavpackparse.c
+++ b/ext/wavpack/gstwavpackparse.c
@@ -57,9 +57,9 @@ static GstStaticPadTemplate wvc_src_factory = GST_STATIC_PAD_TEMPLATE ("wvcsrc",
GST_STATIC_CAPS ("audio/x-wavpack-correction, " "framed = (boolean) true")
);
-static gboolean gst_wavepack_parse_sink_activate (GstPad * sinkpad);
+static gboolean gst_wavpack_parse_sink_activate (GstPad * sinkpad);
static gboolean
-gst_wavepack_parse_sink_activate_pull (GstPad * sinkpad, gboolean active);
+gst_wavpack_parse_sink_activate_pull (GstPad * sinkpad, gboolean active);
static void gst_wavpack_parse_loop (GstElement * element);
static GstStateChangeReturn gst_wavpack_parse_change_state (GstElement *
@@ -68,6 +68,7 @@ static void gst_wavpack_parse_reset (GstWavpackParse * wavpackparse);
static gint64 gst_wavpack_parse_get_upstream_length (GstWavpackParse * wvparse);
static GstBuffer *gst_wavpack_parse_pull_buffer (GstWavpackParse * wvparse,
gint64 offset, guint size, GstFlowReturn * flow);
+static GstFlowReturn gst_wavpack_parse_chain (GstPad * pad, GstBuffer * buf);
GST_BOILERPLATE (GstWavpackParse, gst_wavpack_parse, GstElement,
GST_TYPE_ELEMENT);
@@ -108,7 +109,7 @@ gst_wavpack_parse_class_init (GstWavpackParseClass * klass)
gobject_class = (GObjectClass *) klass;
gstelement_class = (GstElementClass *) klass;
- gobject_class->finalize = gst_wavpack_parse_finalize;
+ gobject_class->finalize = GST_DEBUG_FUNCPTR (gst_wavpack_parse_finalize);
gstelement_class->change_state =
GST_DEBUG_FUNCPTR (gst_wavpack_parse_change_state);
}
@@ -188,7 +189,7 @@ gst_wavpack_parse_index_append_entry (GstWavpackParse * wvparse,
static void
gst_wavpack_parse_reset (GstWavpackParse * wavpackparse)
{
- wavpackparse->total_samples = 0;
+ wavpackparse->total_samples = -1;
wavpackparse->samplerate = 0;
wavpackparse->channels = 0;
@@ -203,6 +204,12 @@ gst_wavpack_parse_reset (GstWavpackParse * wavpackparse)
wavpackparse->entries = NULL;
}
+ if (wavpackparse->adapter) {
+ gst_adapter_clear (wavpackparse->adapter);
+ g_object_unref (wavpackparse->adapter);
+ wavpackparse->adapter = NULL;
+ }
+
if (wavpackparse->srcpad != NULL) {
gboolean res;
@@ -220,6 +227,19 @@ gst_wavpack_parse_reset (GstWavpackParse * wavpackparse)
wavpackparse->queued_events = NULL;
}
+static const GstQueryType *
+gst_wavpack_parse_get_src_query_types (GstPad * pad)
+{
+ static const GstQueryType types[] = {
+ GST_QUERY_POSITION,
+ GST_QUERY_DURATION,
+ GST_QUERY_SEEKING,
+ 0
+ };
+
+ return types;
+}
+
static gboolean
gst_wavpack_parse_src_query (GstPad * pad, GstQuery * query)
{
@@ -268,10 +288,12 @@ gst_wavpack_parse_src_query (GstPad * pad, GstQuery * query)
GST_OBJECT_LOCK (wavpackparse);
rate = wavpackparse->samplerate;
- len = wavpackparse->total_samples;
+ /* FIXME: return 0 if we work in push based mode to let totem
+ * recognize that we can't seek */
+ len = (wavpackparse->adapter) ? 0 : wavpackparse->total_samples;
GST_OBJECT_UNLOCK (wavpackparse);
- if (len <= 0 || rate == 0) {
+ if (len < 0 || rate == 0) {
GST_DEBUG_OBJECT (wavpackparse, "haven't read header yet");
break;
}
@@ -295,6 +317,24 @@ gst_wavpack_parse_src_query (GstPad * pad, GstQuery * query)
}
break;
}
+ case GST_QUERY_SEEKING:{
+ gst_query_parse_seeking (query, &format, NULL, NULL, NULL);
+ if (format == GST_FORMAT_TIME || format == GST_FORMAT_DEFAULT) {
+ gboolean seekable;
+ gint64 duration = -1;
+
+ gst_pad_query_duration (pad, &format, &duration);
+
+ /* can't seek in streaming mode yet */
+ GST_OBJECT_LOCK (wavpackparse);
+ seekable = (wavpackparse->adapter != NULL);
+ GST_OBJECT_UNLOCK (wavpackparse);
+
+ gst_query_set_seeking (query, GST_FORMAT_TIME, seekable, 0, duration);
+ ret = TRUE;
+ }
+ break;
+ }
default:{
ret = gst_pad_query_default (pad, query);
break;
@@ -433,6 +473,11 @@ gst_wavpack_parse_handle_seek_event (GstWavpackParse * wvparse,
gint64 chunk_start; /* first sample in chunk we seek to */
guint rate;
+ if (wvparse->adapter) {
+ GST_DEBUG_OBJECT (wvparse, "seeking in streaming mode not implemented yet");
+ return FALSE;
+ }
+
gst_event_parse_seek (event, &speed, &format, &seek_flags, &start_type,
&start, &stop_type, &stop);
@@ -544,14 +589,41 @@ gst_wavpack_parse_sink_event (GstPad * pad, GstEvent * event)
parse = GST_WAVPACK_PARSE (gst_pad_get_parent (pad));
- /* stream lock is recursive, should be fine for all events */
- GST_PAD_STREAM_LOCK (pad);
- if (parse->srcpad == NULL) {
- parse->queued_events = g_list_append (parse->queued_events, event);
- } else {
- ret = gst_pad_push_event (parse->srcpad, event);
+ switch (GST_EVENT_TYPE (event)) {
+ case GST_EVENT_FLUSH_STOP:{
+ if (parse->adapter) {
+ gst_adapter_clear (parse->adapter);
+ }
+ ret = gst_pad_push_event (pad, event);
+ break;
+ }
+ case GST_EVENT_NEWSEGMENT:{
+ parse->need_newsegment = TRUE;
+ gst_event_unref (event);
+ ret = TRUE;
+ break;
+ }
+ case GST_EVENT_EOS:{
+ if (parse->adapter) {
+ /* remove all bytes that are left in the adapter after EOS. They can't
+ * be a complete Wavpack block and we can't do anything with them */
+ gst_adapter_clear (parse->adapter);
+ }
+ ret = gst_pad_push_event (pad, event);
+ break;
+ }
+ default:{
+ /* stream lock is recursive, should be fine for all events */
+ GST_PAD_STREAM_LOCK (pad);
+ if (parse->srcpad == NULL) {
+ parse->queued_events = g_list_append (parse->queued_events, event);
+ } else {
+ ret = gst_pad_push_event (parse->srcpad, event);
+ }
+ GST_PAD_STREAM_UNLOCK (pad);
+ }
}
- GST_PAD_STREAM_UNLOCK (pad);
+
gst_object_unref (parse);
return ret;
@@ -589,11 +661,13 @@ gst_wavpack_parse_init (GstWavpackParse * wavpackparse,
wavpackparse->sinkpad = gst_pad_new_from_template (tmpl, "sink");
gst_pad_set_activate_function (wavpackparse->sinkpad,
- GST_DEBUG_FUNCPTR (gst_wavepack_parse_sink_activate));
+ GST_DEBUG_FUNCPTR (gst_wavpack_parse_sink_activate));
gst_pad_set_activatepull_function (wavpackparse->sinkpad,
- GST_DEBUG_FUNCPTR (gst_wavepack_parse_sink_activate_pull));
+ GST_DEBUG_FUNCPTR (gst_wavpack_parse_sink_activate_pull));
gst_pad_set_event_function (wavpackparse->sinkpad,
GST_DEBUG_FUNCPTR (gst_wavpack_parse_sink_event));
+ gst_pad_set_chain_function (wavpackparse->sinkpad,
+ GST_DEBUG_FUNCPTR (gst_wavpack_parse_chain));
gst_element_add_pad (GST_ELEMENT (wavpackparse), wavpackparse->sinkpad);
@@ -738,6 +812,8 @@ gst_wavpack_parse_create_src_pad (GstWavpackParse * wvparse, GstBuffer * buf,
gst_pad_set_query_function (wvparse->srcpad,
GST_DEBUG_FUNCPTR (gst_wavpack_parse_src_query));
+ gst_pad_set_query_type_function (wvparse->srcpad,
+ GST_DEBUG_FUNCPTR (gst_wavpack_parse_get_src_query_types));
gst_pad_set_event_function (wvparse->srcpad,
GST_DEBUG_FUNCPTR (gst_wavpack_parse_src_event));
@@ -752,6 +828,43 @@ gst_wavpack_parse_create_src_pad (GstWavpackParse * wvparse, GstBuffer * buf,
return TRUE;
}
+static GstFlowReturn
+gst_wavpack_parse_push_buffer (GstWavpackParse * wvparse, GstBuffer * buf,
+ WavpackHeader * header)
+{
+ wvparse->current_offset += header->ckSize + 8;
+
+ wvparse->segment.last_stop = header->block_index;
+
+ if (wvparse->need_newsegment) {
+ if (gst_wavpack_parse_send_newsegment (wvparse, FALSE))
+ wvparse->need_newsegment = FALSE;
+ }
+
+ /* send any queued events */
+ if (wvparse->queued_events) {
+ GList *l;
+
+ for (l = wvparse->queued_events; l != NULL; l = l->next) {
+ gst_pad_push_event (wvparse->srcpad, GST_EVENT (l->data));
+ }
+ g_list_free (wvparse->queued_events);
+ wvparse->queued_events = NULL;
+ }
+
+ GST_BUFFER_TIMESTAMP (buf) = gst_util_uint64_scale_int (header->block_index,
+ GST_SECOND, wvparse->samplerate);
+ GST_BUFFER_DURATION (buf) = gst_util_uint64_scale_int (header->block_samples,
+ GST_SECOND, wvparse->samplerate);
+ GST_BUFFER_OFFSET (buf) = header->block_index;
+ gst_buffer_set_caps (buf, GST_PAD_CAPS (wvparse->srcpad));
+
+ GST_LOG_OBJECT (wvparse, "Pushing buffer with time %" GST_TIME_FORMAT,
+ GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (buf)));
+
+ return gst_pad_push (wvparse->srcpad, buf);
+}
+
static void
gst_wavpack_parse_loop (GstElement * element)
{
@@ -797,37 +910,7 @@ gst_wavpack_parse_loop (GstElement * element)
gst_wavpack_parse_index_append_entry (wavpackparse,
wavpackparse->current_offset, header.block_index, header.block_samples);
- wavpackparse->current_offset += header.ckSize + 8;
-
- wavpackparse->segment.last_stop = header.block_index;
-
- if (wavpackparse->need_newsegment) {
- if (gst_wavpack_parse_send_newsegment (wavpackparse, FALSE))
- wavpackparse->need_newsegment = FALSE;
- }
-
- /* send any queued events */
- if (wavpackparse->queued_events) {
- GList *l;
-
- for (l = wavpackparse->queued_events; l != NULL; l = l->next) {
- gst_pad_push_event (wavpackparse->srcpad, GST_EVENT (l->data));
- }
- g_list_free (wavpackparse->queued_events);
- wavpackparse->queued_events = NULL;
- }
-
- GST_BUFFER_TIMESTAMP (buf) = gst_util_uint64_scale_int (header.block_index,
- GST_SECOND, wavpackparse->samplerate);
- GST_BUFFER_DURATION (buf) = gst_util_uint64_scale_int (header.block_samples,
- GST_SECOND, wavpackparse->samplerate);
- GST_BUFFER_OFFSET (buf) = header.block_index;
- gst_buffer_set_caps (buf, GST_PAD_CAPS (wavpackparse->srcpad));
-
- GST_LOG_OBJECT (wavpackparse, "Pushing buffer with time %" GST_TIME_FORMAT,
- GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (buf)));
-
- flow_ret = gst_pad_push (wavpackparse->srcpad, buf);
+ flow_ret = gst_wavpack_parse_push_buffer (wavpackparse, buf, &header);
if (flow_ret != GST_FLOW_OK) {
GST_DEBUG_OBJECT (wavpackparse, "Push failed, flow: %s",
gst_flow_get_name (flow_ret));
@@ -852,6 +935,61 @@ pause:
}
}
+static GstFlowReturn
+gst_wavpack_parse_chain (GstPad * pad, GstBuffer * buf)
+{
+ GstWavpackParse *wvparse = GST_WAVPACK_PARSE (GST_PAD_PARENT (pad));
+ GstFlowReturn ret = GST_FLOW_OK;
+ WavpackHeader wph;
+ const guint8 *tmp_buf;
+
+ if (!wvparse->adapter) {
+ wvparse->adapter = gst_adapter_new ();
+ }
+
+ if (GST_BUFFER_FLAG_IS_SET (buf, GST_BUFFER_FLAG_DISCONT)) {
+ gst_adapter_clear (wvparse->adapter);
+ }
+
+ gst_adapter_push (wvparse->adapter, buf);
+
+ if (gst_adapter_available (wvparse->adapter) < sizeof (WavpackHeader))
+ return ret;
+
+ tmp_buf = gst_adapter_peek (wvparse->adapter, sizeof (WavpackHeader));
+ gst_wavpack_read_header (&wph, (guint8 *) tmp_buf);
+
+ /* FIXME: should check for wavpack marker here and re-sync if not */
+
+ while (gst_adapter_available (wvparse->adapter) >= wph.ckSize + 4 * 1 + 4) {
+ GstBuffer *outbuf =
+ gst_adapter_take_buffer (wvparse->adapter, wph.ckSize + 4 * 1 + 4);
+
+ if (!outbuf)
+ return GST_FLOW_ERROR;
+
+ if (wvparse->srcpad == NULL) {
+ if (!gst_wavpack_parse_create_src_pad (wvparse, outbuf, &wph)) {
+ GST_ELEMENT_ERROR (wvparse, STREAM, DECODE, (NULL), (NULL));
+ ret = GST_FLOW_ERROR;
+ break;
+ }
+ }
+
+ ret = gst_wavpack_parse_push_buffer (wvparse, outbuf, &wph);
+
+ if (ret != GST_FLOW_OK)
+ break;
+
+ if (gst_adapter_available (wvparse->adapter) >= sizeof (WavpackHeader)) {
+ tmp_buf = gst_adapter_peek (wvparse->adapter, sizeof (WavpackHeader));
+ gst_wavpack_read_header (&wph, (guint8 *) tmp_buf);
+ }
+ }
+
+ return ret;
+}
+
static GstStateChangeReturn
gst_wavpack_parse_change_state (GstElement * element, GstStateChange transition)
{
@@ -880,19 +1018,18 @@ gst_wavpack_parse_change_state (GstElement * element, GstStateChange transition)
return ret;
}
-
static gboolean
-gst_wavepack_parse_sink_activate (GstPad * sinkpad)
+gst_wavpack_parse_sink_activate (GstPad * sinkpad)
{
if (gst_pad_check_pull_range (sinkpad)) {
return gst_pad_activate_pull (sinkpad, TRUE);
} else {
- return FALSE;
+ return gst_pad_activate_push (sinkpad, TRUE);
}
}
static gboolean
-gst_wavepack_parse_sink_activate_pull (GstPad * sinkpad, gboolean active)
+gst_wavpack_parse_sink_activate_pull (GstPad * sinkpad, gboolean active)
{
gboolean result;
diff --git a/ext/wavpack/gstwavpackparse.h b/ext/wavpack/gstwavpackparse.h
index 37321d83..d849da74 100644
--- a/ext/wavpack/gstwavpackparse.h
+++ b/ext/wavpack/gstwavpackparse.h
@@ -23,6 +23,7 @@
#define __GST_WAVPACK_PARSE_H__
#include <gst/gst.h>
+#include <gst/base/gstadapter.h>
G_BEGIN_DECLS
@@ -67,6 +68,8 @@ struct _GstWavpackParse
GstSegment segment; /* the currently configured segment, in
* samples/audio frames (DEFAULT format) */
+ GstAdapter *adapter; /* when operating chain-based, otherwise NULL */
+
/* Array of GstWavpackParseIndexEntry structs, mapping known
* sample offsets to byte offsets. Is kept increasing without
* gaps (ie. append only and consecutive entries must always