summaryrefslogtreecommitdiffstats
path: root/ext/wavpack/gstwavpackparse.c
diff options
context:
space:
mode:
Diffstat (limited to 'ext/wavpack/gstwavpackparse.c')
-rw-r--r--ext/wavpack/gstwavpackparse.c972
1 files changed, 561 insertions, 411 deletions
diff --git a/ext/wavpack/gstwavpackparse.c b/ext/wavpack/gstwavpackparse.c
index 5621d4e6..f00e337e 100644
--- a/ext/wavpack/gstwavpackparse.c
+++ b/ext/wavpack/gstwavpackparse.c
@@ -1,5 +1,6 @@
/* GStreamer wavpack plugin
* (c) 2005 Arwed v. Merkatz <v.merkatz@gmx.net>
+ * (c) 2006 Tim-Philipp Müller <tim centricular net>
*
* gstwavpackparse.c: wavpack file parser
*
@@ -31,17 +32,6 @@
GST_DEBUG_CATEGORY_STATIC (gst_wavpack_parse_debug);
#define GST_CAT_DEFAULT gst_wavpack_parse_debug
-/* Filter signals and args */
-enum
-{
- LAST_SIGNAL
-};
-
-enum
-{
- ARG_0
-};
-
static GstStaticPadTemplate sink_factory = GST_STATIC_PAD_TEMPLATE ("sink",
GST_PAD_SINK,
GST_PAD_ALWAYS,
@@ -54,7 +44,7 @@ static GstStaticPadTemplate src_factory = GST_STATIC_PAD_TEMPLATE ("src",
GST_PAD_SRC,
GST_PAD_SOMETIMES,
GST_STATIC_CAPS ("audio/x-wavpack, "
- "width = (int) { 8, 16, 24 }, "
+ "width = (int) { 8, 16, 24, 32 }, "
"channels = (int) { 1, 2 }, "
"rate = (int) [ 6000, 192000 ], " "framed = (boolean) true")
);
@@ -65,47 +55,22 @@ static GstStaticPadTemplate wvc_src_factory = GST_STATIC_PAD_TEMPLATE ("wvcsrc",
GST_STATIC_CAPS ("audio/x-wavpack-correction, " "framed = (boolean) true")
);
-static void gst_wavpack_parse_class_init (GstWavpackParseClass * klass);
-static void gst_wavpack_parse_base_init (GstWavpackParseClass * klass);
-static void gst_wavpack_parse_init (GstWavpackParse * wavpackparse);
-
static gboolean gst_wavepack_parse_sink_activate (GstPad * sinkpad);
static gboolean
gst_wavepack_parse_sink_activate_pull (GstPad * sinkpad, gboolean active);
-static gboolean gst_wavpack_parse_sink_event (GstPad * pad, GstEvent * event);
-
static void gst_wavpack_parse_loop (GstElement * element);
static GstStateChangeReturn gst_wavpack_parse_change_state (GstElement *
element, GstStateChange transition);
+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 GstElementClass *parent = NULL;
-
-GType
-gst_wavpack_parse_get_type (void)
-{
- static GType plugin_type = 0;
-
- if (!plugin_type) {
- static const GTypeInfo plugin_info = {
- sizeof (GstWavpackParseClass),
- (GBaseInitFunc) gst_wavpack_parse_base_init,
- NULL,
- (GClassInitFunc) gst_wavpack_parse_class_init,
- NULL,
- NULL,
- sizeof (GstWavpackParse),
- 0,
- (GInstanceInitFunc) gst_wavpack_parse_init,
- };
- plugin_type = g_type_register_static (GST_TYPE_ELEMENT,
- "GstWavpackParse", &plugin_info, 0);
- }
- return plugin_type;
-}
+GST_BOILERPLATE (GstWavpackParse, gst_wavpack_parse, GstElement,
+ GST_TYPE_ELEMENT)
-static void
-gst_wavpack_parse_base_init (GstWavpackParseClass * klass)
+ static void gst_wavpack_parse_base_init (gpointer klass)
{
static GstElementDetails plugin_details = {
"Wavpack file parser",
@@ -127,7 +92,8 @@ gst_wavpack_parse_base_init (GstWavpackParseClass * klass)
static void
gst_wavpack_parse_dispose (GObject * object)
{
- G_OBJECT_CLASS (parent)->dispose (object);
+ gst_wavpack_parse_reset (GST_WAVPACK_PARSE (object));
+ G_OBJECT_CLASS (parent_class)->dispose (object);
}
static void
@@ -139,513 +105,703 @@ gst_wavpack_parse_class_init (GstWavpackParseClass * klass)
gobject_class = (GObjectClass *) klass;
gstelement_class = (GstElementClass *) klass;
- parent = g_type_class_ref (GST_TYPE_ELEMENT);
-
gobject_class->dispose = gst_wavpack_parse_dispose;
- gstelement_class->change_state = gst_wavpack_parse_change_state;
+ gstelement_class->change_state =
+ GST_DEBUG_FUNCPTR (gst_wavpack_parse_change_state);
+}
+
+static GstWavpackParseIndexEntry *
+gst_wavpack_parse_index_get_last_entry (GstWavpackParse * wvparse)
+{
+ gint last;
+
+ g_assert (wvparse->entries != NULL);
+ g_assert (wvparse->entries->len > 0);
+
+ last = wvparse->entries->len - 1;
+ return &g_array_index (wvparse->entries, GstWavpackParseIndexEntry, last);
+}
+
+static GstWavpackParseIndexEntry *
+gst_wavpack_parse_index_get_entry_from_sample (GstWavpackParse * wvparse,
+ gint64 sample_offset)
+{
+ gint i;
+
+ if (wvparse->entries == NULL || wvparse->entries->len == 0)
+ return NULL;
+
+ for (i = wvparse->entries->len - 1; i >= 0; --i) {
+ GstWavpackParseIndexEntry *entry;
+
+ entry = &g_array_index (wvparse->entries, GstWavpackParseIndexEntry, i);
+
+ GST_LOG_OBJECT (wvparse, "Index entry %03u: sample %" G_GINT64_FORMAT " @"
+ " byte %" G_GINT64_FORMAT, entry->sample_offset, entry->byte_offset);
+
+ if (entry->sample_offset <= sample_offset &&
+ sample_offset < entry->sample_offset_end) {
+ GST_LOG_OBJECT (wvparse, "found match");
+ return entry;
+ }
+ }
+ GST_LOG_OBJECT (wvparse, "no match in index");
+ return NULL;
+}
+
+static void
+gst_wavpack_parse_index_append_entry (GstWavpackParse * wvparse,
+ gint64 byte_offset, gint64 sample_offset, gint64 num_samples)
+{
+ GstWavpackParseIndexEntry entry;
+
+ if (wvparse->entries == NULL) {
+ wvparse->entries = g_array_new (FALSE, TRUE,
+ sizeof (GstWavpackParseIndexEntry));
+ } else {
+ /* do we have this one already? */
+ entry = *gst_wavpack_parse_index_get_last_entry (wvparse);
+ if (entry.byte_offset >= byte_offset)
+ return;
+ }
+
+ GST_LOG_OBJECT (wvparse, "Adding index entry %8" G_GINT64_FORMAT " - %"
+ GST_TIME_FORMAT " @ offset 0x%08" G_GINT64_MODIFIER "x", sample_offset,
+ GST_TIME_ARGS (gst_util_uint64_scale_int (sample_offset,
+ GST_SECOND, wvparse->samplerate)), byte_offset);
+
+ entry.byte_offset = byte_offset;
+ entry.sample_offset = sample_offset;
+ entry.sample_offset_end = sample_offset + num_samples;
+ g_array_append_val (wvparse->entries, entry);
+}
+
+static void
+gst_wavpack_parse_reset (GstWavpackParse * wavpackparse)
+{
+ wavpackparse->total_samples = 0;
+ wavpackparse->samplerate = 0;
+ wavpackparse->channels = 0;
+
+ gst_segment_init (&wavpackparse->segment, GST_FORMAT_UNDEFINED);
+
+ wavpackparse->current_offset = 0;
+ wavpackparse->need_newsegment = TRUE;
+ wavpackparse->upstream_length = -1;
+
+ if (wavpackparse->entries) {
+ g_array_free (wavpackparse->entries, TRUE);
+ wavpackparse->entries = NULL;
+ }
+
+ if (wavpackparse->srcpad != NULL) {
+ gboolean res;
+
+ GST_DEBUG_OBJECT (wavpackparse, "Removing src pad");
+ res = gst_element_remove_pad (GST_ELEMENT (wavpackparse),
+ wavpackparse->srcpad);
+ g_return_if_fail (res != FALSE);
+ gst_object_unref (wavpackparse->srcpad);
+ wavpackparse->srcpad = NULL;
+ }
}
static gboolean
gst_wavpack_parse_src_query (GstPad * pad, GstQuery * query)
{
GstWavpackParse *wavpackparse = GST_WAVPACK_PARSE (gst_pad_get_parent (pad));
- GstFormat format = GST_FORMAT_DEFAULT;
- gint64 value;
+ GstFormat format;
gboolean ret = FALSE;
switch (GST_QUERY_TYPE (query)) {
- case GST_QUERY_POSITION:
- gst_query_parse_position (query, &format, &value);
- if (format == GST_FORMAT_TIME) {
- value = wavpackparse->timestamp;
- gst_query_set_duration (query, format, value);
- gst_object_unref (wavpackparse);
- ret = TRUE;
+ case GST_QUERY_POSITION:{
+ gint64 cur, len;
+ guint rate;
+
+ GST_OBJECT_LOCK (wavpackparse);
+ cur = wavpackparse->segment.last_stop;
+ len = wavpackparse->total_samples;
+ rate = wavpackparse->samplerate;
+ GST_OBJECT_UNLOCK (wavpackparse);
+
+ if (len <= 0 || rate == 0) {
+ GST_DEBUG_OBJECT (wavpackparse, "haven't read header yet");
break;
}
- break;
- case GST_QUERY_DURATION:
- gst_query_parse_duration (query, &format, &value);
-
- if (format == GST_FORMAT_TIME) {
- if (wavpackparse->total_samples == 0) {
- value = 0;
- gst_query_set_duration (query, format, value);
- gst_object_unref (wavpackparse);
- ret = FALSE;
+
+ gst_query_parse_position (query, &format, NULL);
+
+ switch (format) {
+ case GST_FORMAT_TIME:
+ cur = gst_util_uint64_scale_int (cur, GST_SECOND, rate);
+ gst_query_set_position (query, GST_FORMAT_TIME, cur);
+ ret = TRUE;
+ break;
+ case GST_FORMAT_DEFAULT:
+ gst_query_set_position (query, GST_FORMAT_DEFAULT, cur);
+ ret = TRUE;
break;
- }
- value = ((gdouble) wavpackparse->total_samples /
- (gdouble) wavpackparse->samplerate) * GST_SECOND;
- gst_query_set_duration (query, format, value);
- gst_object_unref (wavpackparse);
- ret = TRUE;
+ default:
+ GST_DEBUG_OBJECT (wavpackparse, "cannot handle position query in "
+ "%s format", gst_format_get_name (format));
+ break;
+ }
+ break;
+ }
+ case GST_QUERY_DURATION:{
+ gint64 len;
+ guint rate;
+
+ GST_OBJECT_LOCK (wavpackparse);
+ rate = wavpackparse->samplerate;
+ len = wavpackparse->total_samples;
+ GST_OBJECT_UNLOCK (wavpackparse);
+
+ if (len <= 0 || rate == 0) {
+ GST_DEBUG_OBJECT (wavpackparse, "haven't read header yet");
break;
}
+
+ gst_query_parse_duration (query, &format, NULL);
+
+ switch (format) {
+ case GST_FORMAT_TIME:
+ len = gst_util_uint64_scale_int (len, GST_SECOND, rate);
+ gst_query_set_duration (query, GST_FORMAT_TIME, len);
+ ret = TRUE;
+ break;
+ case GST_FORMAT_DEFAULT:
+ gst_query_set_duration (query, GST_FORMAT_DEFAULT, len);
+ ret = TRUE;
+ break;
+ default:
+ GST_DEBUG_OBJECT (wavpackparse, "cannot handle duration query in "
+ "%s format", gst_format_get_name (format));
+ break;
+ }
break;
- default:
- gst_object_unref (wavpackparse);
+ }
+ default:{
ret = gst_pad_query_default (pad, query);
break;
+ }
}
+ gst_object_unref (wavpackparse);
return ret;
}
+/* returns TRUE on success, with byte_offset set to the offset of the
+ * wavpack chunk containing the sample requested. start_sample will be
+ * set to the first sample in the chunk starting at byte_offset.
+ * Scanning from the last known header offset to the wanted position
+ * when seeking forward isn't very clever, but seems fast enough in
+ * practice and has the nice side effect of populating our index
+ * table */
static gboolean
-gst_wavpack_parse_src_event (GstPad * pad, GstEvent * event)
+gst_wavpack_parse_scan_to_find_sample (GstWavpackParse * parse,
+ gint64 sample, gint64 * byte_offset, gint64 * start_sample)
{
- GstWavpackParse *wavpackparse;
- GstSeekType type;
- GstFormat format;
- gboolean need_flush;
- gint64 offset, dest;
- GstSeekFlags flags;
- gboolean ret = TRUE;
+ GstWavpackParseIndexEntry *entry;
+ GstFlowReturn ret;
+ gint64 off = 0;
+
+ /* first, check if we have to scan at all */
+ entry = gst_wavpack_parse_index_get_entry_from_sample (parse, sample);
+ if (entry) {
+ *byte_offset = entry->byte_offset;
+ *start_sample = entry->sample_offset;
+ GST_LOG_OBJECT (parse, "Found index entry: sample %" G_GINT64_FORMAT
+ " @ offset %" G_GINT64_FORMAT, entry->sample_offset,
+ entry->byte_offset);
+ return TRUE;
+ }
- wavpackparse = GST_WAVPACK_PARSE (gst_pad_get_parent (pad));
+ GST_LOG_OBJECT (parse, "No matching entry in index, scanning file ...");
- if (GST_EVENT_TYPE (event) != GST_EVENT_SEEK) {
- GstPad *peer;
+ /* if we have an index, we can start scanning from the last known offset
+ * in there, after all we know our wanted sample is not in the index */
+ if (parse->entries && parse->entries->len > 0) {
+ GstWavpackParseIndexEntry *entry;
- if (!(peer = gst_pad_get_peer (wavpackparse->sinkpad))) {
- ret = FALSE;
- goto done;
- }
- ret = gst_pad_send_event (peer, event);
- gst_object_unref (peer);
- goto done;
+ entry = gst_wavpack_parse_index_get_last_entry (parse);
+ off = entry->byte_offset;
}
- gst_event_parse_seek (event, NULL, &format, &flags, &type, &offset, NULL,
- NULL);
+ /* now scan forward until we find the chunk we're looking for or hit EOS */
+ do {
+ WavpackHeader header = { {0,}
+ , 0, };
+ GstBuffer *buf;
- need_flush = flags & GST_SEEK_FLAG_FLUSH;
+ buf = gst_wavpack_parse_pull_buffer (parse, off, sizeof (WavpackHeader),
+ &ret);
+ if (buf == NULL)
+ break;
+ gst_wavpack_read_header (&header, GST_BUFFER_DATA (buf));
+ gst_buffer_unref (buf);
- if (offset < 0 || type != GST_SEEK_TYPE_SET) {
- ret = FALSE;
- goto done;
- }
+ gst_wavpack_parse_index_append_entry (parse, off, header.block_index,
+ header.block_samples);
- if (format == GST_FORMAT_TIME) {
- dest = offset * wavpackparse->samplerate / GST_SECOND;
- } else if (format == GST_FORMAT_DEFAULT) {
- dest = offset;
- } else {
- ret = FALSE;
- goto done;
- }
+ if (header.block_index <= sample &&
+ sample < (header.block_index + header.block_samples)) {
+ *byte_offset = off;
+ *start_sample = header.block_index;
+ return TRUE;
+ }
- wavpackparse->need_discont = TRUE;
- wavpackparse->need_flush = need_flush;
+ off += header.ckSize + 8;
+ } while (1);
- wavpackparse->seek_pending = TRUE;
- wavpackparse->seek_offset = dest;
+ GST_DEBUG_OBJECT (parse, "scan failed: %s (off=0x%08" G_GINT64_MODIFIER "x)",
+ gst_flow_get_name (ret), off);
-done:
- gst_event_unref (event);
- gst_object_unref (wavpackparse);
- return ret;
+ return FALSE;
}
-#define BUFSIZE 4096
-
-static guint64
-find_header (GstWavpackParse * wavpackparse, guint64 filepos,
- WavpackHeader * wphdr)
+static gboolean
+gst_wavpack_parse_send_newsegment (GstWavpackParse * wvparse, gboolean update)
{
- guint64 pos = filepos;
- gint read = 0;
- GstBuffer *buf = NULL;
-
- while (TRUE) {
- guint8 *cur;
-
- if (GST_FLOW_OK != gst_pad_pull_range (wavpackparse->sinkpad,
- wavpackparse->flushed_bytes + filepos, BUFSIZE, &buf)) {
- wavpackparse->eos = TRUE;
- gst_pad_push_event (wavpackparse->srcpad, gst_event_new_eos ());
- return 0;
- }
- read = GST_BUFFER_SIZE (buf);
-
- if (read == 0) {
- wavpackparse->eos = TRUE;
- gst_pad_push_event (wavpackparse->srcpad, gst_event_new_eos ());
- gst_buffer_unref (buf);
- return 0;
- }
+ GstSegment *s = &wvparse->segment;
+ gboolean ret;
+ gint64 stop_time = -1;
+ gint64 start_time = 0;
+ gint64 cur_pos_time;
+ gint64 diff;
+
+ /* segment is in DEFAULT format, but we want to send a TIME newsegment */
+ start_time = gst_util_uint64_scale_int (s->start, GST_SECOND,
+ wvparse->samplerate);
+
+ if (s->stop != -1) {
+ stop_time = gst_util_uint64_scale_int (s->stop, GST_SECOND,
+ wvparse->samplerate);
+ }
- cur = GST_BUFFER_DATA (buf);
- do {
- if (cur[0] == 'w' && cur[1] == 'v' && cur[2] == 'p' && cur[3] == 'k') {
- gst_wavpack_read_header (wphdr, cur);
- pos += (cur - GST_BUFFER_DATA (buf));
- gst_buffer_unref (buf);
- buf = NULL;
- return pos;
- }
- cur++;
- } while ((cur - GST_BUFFER_DATA (buf)) <
- (BUFSIZE - sizeof (WavpackHeader)));
+ GST_DEBUG_OBJECT (wvparse, "sending newsegment from %" GST_TIME_FORMAT
+ " to %" GST_TIME_FORMAT, GST_TIME_ARGS (start_time),
+ GST_TIME_ARGS (stop_time));
- wavpackparse->flushed_bytes += BUFSIZE - sizeof (WavpackHeader);
+ /* after a seek, s->last_stop will point to a chunk boundary, ie. from
+ * which sample we will start sending data again, while s->start will
+ * point to the sample we actually want to seek to and want to start
+ * playing right after the seek. Adjust clock-time for the difference
+ * so we start playing from start_time */
+ cur_pos_time = gst_util_uint64_scale_int (s->last_stop, GST_SECOND,
+ wvparse->samplerate);
+ diff = start_time - cur_pos_time;
- pos += BUFSIZE - sizeof (WavpackHeader);
- }
+ ret = gst_pad_push_event (wvparse->srcpad,
+ gst_event_new_new_segment (update, s->rate, GST_FORMAT_TIME,
+ start_time, stop_time, start_time - diff));
- /* never reached */
-
- if (buf) {
- gst_buffer_unref (buf);
- buf = NULL;
- }
- return wavpackparse->duration - wavpackparse->flushed_bytes;
+ return ret;
}
-/* find the position of sample in the input bytestream, adapted from the
- * original wavpack find_sample function */
-static guint64
-find_sample (GstWavpackParse * wavpackparse, guint32 sample)
+static gboolean
+gst_wavpack_parse_handle_seek_event (GstWavpackParse * wvparse,
+ GstEvent * event)
{
- WavpackHeader wphdr;
- guint64 file_pos1 = 0;
- guint64 file_pos2 = wavpackparse->duration - wavpackparse->flushed_bytes;
- guint64 sample_pos1 = 0, sample_pos2 = wavpackparse->total_samples;
- double ratio = 0.96;
- int file_skip = 0;
-
- if (sample >= wavpackparse->total_samples) {
- return wavpackparse->duration - wavpackparse->flushed_bytes;
- }
-
- while (1) {
- double bytes_per_sample;
- guint64 seek_pos;
-
- bytes_per_sample = file_pos2 - file_pos1;
- bytes_per_sample /= sample_pos2 - sample_pos1;
- seek_pos = file_pos1 + (file_skip ? 32 : 0);
- seek_pos += (guint64) (bytes_per_sample * (sample - sample_pos1) * ratio);
- seek_pos = find_header (wavpackparse, seek_pos, &wphdr);
-
- if (seek_pos == wavpackparse->duration - wavpackparse->flushed_bytes
- || seek_pos >= file_pos2) {
- if (ratio > 0.0) {
- if ((ratio -= 0.24) < 0.0)
- ratio = 0.0;
- } else {
- return wavpackparse->duration - wavpackparse->flushed_bytes;
- }
- } else if (wphdr.block_index > sample) {
- sample_pos2 = wphdr.block_index;
- file_pos2 = seek_pos;
- } else if (wphdr.block_index + wphdr.block_samples <= sample) {
- if (seek_pos == file_pos1)
- file_skip = 1;
- else {
- sample_pos1 = wphdr.block_index;
- file_pos1 = seek_pos;
- }
- } else {
- return seek_pos;
- }
+ GstSeekFlags seek_flags;
+ GstSeekType start_type;
+ GstSeekType stop_type;
+ GstSegment segment;
+ GstFormat format;
+ gboolean only_update;
+ gboolean flush, ret;
+ gdouble speed;
+ gint64 stop;
+ gint64 start; /* sample we want to seek to */
+ gint64 byte_offset; /* byte offset the chunk we seek to starts at */
+ gint64 chunk_start; /* first sample in chunk we seek to */
+ guint rate;
+
+ gst_event_parse_seek (event, &speed, &format, &seek_flags, &start_type,
+ &start, &stop_type, &stop);
+
+ if (format != GST_FORMAT_DEFAULT && format != GST_FORMAT_TIME) {
+ GST_DEBUG ("seeking is only supported in TIME or DEFAULT format");
+ return FALSE;
}
-}
+ if (speed < 0.0) {
+ GST_DEBUG ("only forward playback supported, rate %f not allowed", speed);
+ return FALSE;
+ }
-static void
-gst_wavpack_parse_seek (GstWavpackParse * wavpackparse)
-{
- GstBuffer *buf;
- gint num;
- WavpackHeader *header = g_malloc (sizeof (WavpackHeader));
-
- guint64 offset = find_sample (wavpackparse, wavpackparse->seek_offset);
+ GST_OBJECT_LOCK (wvparse);
- if (offset >= wavpackparse->duration - wavpackparse->flushed_bytes) {
- /* seek failed or went beyond the end, go EOS */
- wavpackparse->timestamp =
- ((gdouble) wavpackparse->total_samples /
- (gdouble) wavpackparse->samplerate) * GST_SECOND;
- return;
+ rate = wvparse->samplerate;
+ if (rate == 0) {
+ GST_OBJECT_UNLOCK (wvparse);
+ GST_DEBUG ("haven't read header yet");
+ return FALSE;
}
- if (GST_FLOW_OK != gst_pad_pull_range (wavpackparse->sinkpad,
- wavpackparse->flushed_bytes + offset, sizeof (WavpackHeader), &buf)) {
- wavpackparse->eos = TRUE;
- gst_pad_push_event (wavpackparse->srcpad, gst_event_new_eos ());
- return;
+ /* convert from time to samples if necessary */
+ if (format == GST_FORMAT_TIME) {
+ if (start_type != GST_SEEK_TYPE_NONE)
+ start = gst_util_uint64_scale_int (start, rate, GST_SECOND);
+ if (stop_type != GST_SEEK_TYPE_NONE)
+ stop = gst_util_uint64_scale_int (stop, rate, GST_SECOND);
}
- num = GST_BUFFER_SIZE (buf);
- if (num != sizeof (WavpackHeader)) {
- wavpackparse->eos = TRUE;
- gst_pad_push_event (wavpackparse->srcpad, gst_event_new_eos ());
- return;
+ flush = ((seek_flags & GST_SEEK_FLAG_FLUSH) != 0);
+
+ if (start < 0) {
+ GST_OBJECT_UNLOCK (wvparse);
+ GST_DEBUG_OBJECT (wvparse, "Invalid start sample %" G_GINT64_FORMAT, start);
+ return FALSE;
}
- gst_wavpack_read_header (header, GST_BUFFER_DATA (buf));
- gst_buffer_unref (buf);
+ /* operate on segment copy until we know the seek worked */
+ segment = wvparse->segment;
- if (wavpackparse->need_flush) {
- wavpackparse->need_flush = FALSE;
- gst_pad_push_event (wavpackparse->srcpad, gst_event_new_flush_start ());
- }
+ gst_segment_set_seek (&segment, speed, GST_FORMAT_DEFAULT,
+ seek_flags, start_type, start, stop_type, stop, &only_update);
- wavpackparse->need_discont = TRUE;
- wavpackparse->timestamp =
- ((gdouble) header->block_index / (gdouble) wavpackparse->samplerate) *
- GST_SECOND;
-}
+#if 0
+ if (only_update) {
+ wvparse->segment = segment;
+ gst_wavpack_parse_send_newsegment (wvparse, TRUE);
+ goto done;
+ }
+#endif
-static void
-gst_wavpack_parse_init (GstWavpackParse * wavpackparse)
-{
- GstElementClass *klass = GST_ELEMENT_GET_CLASS (wavpackparse);
+ gst_pad_push_event (wvparse->sinkpad, gst_event_new_flush_start ());
- wavpackparse->duration = -1;
- wavpackparse->flushed_bytes = -1;
- wavpackparse->eos = FALSE;
+ if (flush) {
+ gst_pad_push_event (wvparse->srcpad, gst_event_new_flush_start ());
+ } else {
+ gst_pad_stop_task (wvparse->sinkpad);
+ }
- wavpackparse->sinkpad =
- gst_pad_new_from_template (gst_element_class_get_pad_template (klass,
- "sink"), "sink");
+ GST_PAD_STREAM_LOCK (wvparse->sinkpad);
- gst_pad_set_event_function (wavpackparse->sinkpad,
- gst_wavpack_parse_sink_event);
+ gst_pad_push_event (wvparse->sinkpad, gst_event_new_flush_stop ());
- gst_element_add_pad (GST_ELEMENT (wavpackparse), wavpackparse->sinkpad);
+ if (flush) {
+ gst_pad_push_event (wvparse->srcpad, gst_event_new_flush_stop ());
+ }
- gst_pad_set_activate_function (wavpackparse->sinkpad,
- gst_wavepack_parse_sink_activate);
+ GST_DEBUG_OBJECT (wvparse, "Performing seek to %" GST_TIME_FORMAT " sample %"
+ G_GINT64_FORMAT, GST_TIME_ARGS (segment.start * GST_SECOND / rate),
+ start);
+
+ ret = gst_wavpack_parse_scan_to_find_sample (wvparse, segment.start,
+ &byte_offset, &chunk_start);
+
+ if (ret) {
+ GST_DEBUG_OBJECT (wvparse, "new offset: %" G_GINT64_FORMAT, byte_offset);
+ wvparse->current_offset = byte_offset;
+ /* we want to send a newsegment event with the actual seek position
+ * as start, even though our first buffer might start before the
+ * configured segment. We leave it up to the decoder or sink to crop
+ * the output buffers accordingly */
+ wvparse->segment = segment;
+ wvparse->segment.last_stop = chunk_start;
+ gst_wavpack_parse_send_newsegment (wvparse, FALSE);
+ } else {
+ GST_DEBUG_OBJECT (wvparse, "seek failed: don't know where to seek to");
+ }
- gst_pad_set_activatepull_function (wavpackparse->sinkpad,
- gst_wavepack_parse_sink_activate_pull);
+ GST_PAD_STREAM_UNLOCK (wvparse->sinkpad);
+ GST_OBJECT_UNLOCK (wvparse);
- wavpackparse->srcpad = NULL;
+ gst_pad_start_task (wvparse->sinkpad,
+ (GstTaskFunction) gst_wavpack_parse_loop, wvparse);
+ return ret;
}
static gboolean
-gst_wavpack_parse_sink_event (GstPad * pad, GstEvent * event)
+gst_wavpack_parse_src_event (GstPad * pad, GstEvent * event)
{
- GstWavpackParse *wavpackparse = GST_WAVPACK_PARSE (gst_pad_get_parent (pad));
- gboolean res = TRUE;
+ GstWavpackParse *wavpackparse;
+ gboolean ret;
+
+ wavpackparse = GST_WAVPACK_PARSE (gst_pad_get_parent (pad));
switch (GST_EVENT_TYPE (event)) {
- case GST_EVENT_EOS:
- wavpackparse->eos = TRUE;
- /* fall through */
+ case GST_EVENT_SEEK:
+ ret = gst_wavpack_parse_handle_seek_event (wavpackparse, event);
+ break;
default:
- res = gst_pad_event_default (pad, event);
- gst_object_unref (wavpackparse);
- return res;
+ ret = gst_pad_event_default (pad, event);
break;
}
+ gst_object_unref (wavpackparse);
+ return ret;
}
static void
-gst_wavpack_parse_loop (GstElement * element)
+gst_wavpack_parse_init (GstWavpackParse * wavpackparse,
+ GstWavpackParseClass * gclass)
{
- GstWavpackParse *wavpackparse = GST_WAVPACK_PARSE (element);
- gint num;
- GstBuffer *buf;
- WavpackHeader *header = g_malloc (sizeof (WavpackHeader));
+ GstElementClass *klass = GST_ELEMENT_GET_CLASS (wavpackparse);
+ GstPadTemplate *tmpl;
- GST_PAD_STREAM_LOCK (wavpackparse->sinkpad);
+ tmpl = gst_element_class_get_pad_template (klass, "sink");
+ wavpackparse->sinkpad = gst_pad_new_from_template (tmpl, "sink");
- if (wavpackparse->eos) {
- goto done;
- }
+ gst_pad_set_activate_function (wavpackparse->sinkpad,
+ GST_DEBUG_FUNCPTR (gst_wavepack_parse_sink_activate));
- if (wavpackparse->seek_pending) {
- gst_wavpack_parse_seek (wavpackparse);
- wavpackparse->need_discont = TRUE;
- wavpackparse->seek_pending = FALSE;
- }
+ gst_pad_set_activatepull_function (wavpackparse->sinkpad,
+ GST_DEBUG_FUNCPTR (gst_wavepack_parse_sink_activate_pull));
- if (wavpackparse->need_discont) {
- if (GST_IS_PAD (wavpackparse->srcpad)) {
- gst_pad_push_event (wavpackparse->srcpad,
- gst_event_new_new_segment (FALSE, 1.0, GST_FORMAT_TIME,
- wavpackparse->timestamp, GST_CLOCK_TIME_NONE, 0));
+ gst_element_add_pad (GST_ELEMENT (wavpackparse), wavpackparse->sinkpad);
- wavpackparse->need_discont = FALSE;
+ wavpackparse->srcpad = NULL;
+ gst_wavpack_parse_reset (wavpackparse);
+}
+
+static gint64
+gst_wavpack_parse_get_upstream_length (GstWavpackParse * wavpackparse)
+{
+ GstPad *peer;
+ gint64 length = -1;
+
+ peer = gst_pad_get_peer (wavpackparse->sinkpad);
+ if (peer) {
+ GstFormat format = GST_FORMAT_BYTES;
+
+ if (!gst_pad_query_duration (peer, &format, &length)) {
+ length = -1;
+ } else {
+ GST_DEBUG ("upstream length: %" G_GINT64_FORMAT, length);
}
+ gst_object_unref (peer);
+ } else {
+ GST_DEBUG ("no peer!");
}
- if (GST_FLOW_OK != gst_pad_pull_range (wavpackparse->sinkpad,
- wavpackparse->flushed_bytes, sizeof (WavpackHeader), &buf)) {
- wavpackparse->eos = TRUE;
- gst_pad_push_event (wavpackparse->srcpad, gst_event_new_eos ());
- goto done;
- }
- num = GST_BUFFER_SIZE (buf);
+ return length;
+}
- if (num != sizeof (WavpackHeader)) {
- gst_buffer_unref (buf);
- wavpackparse->eos = TRUE;
- gst_pad_push_event (wavpackparse->srcpad, gst_event_new_eos ());
- goto done;
+static GstBuffer *
+gst_wavpack_parse_pull_buffer (GstWavpackParse * wvparse, gint64 offset,
+ guint size, GstFlowReturn * flow)
+{
+ GstFlowReturn flow_ret;
+ GstBuffer *buf = NULL;
+
+ if (offset + size >= wvparse->upstream_length) {
+ wvparse->upstream_length = gst_wavpack_parse_get_upstream_length (wvparse);
+ if (offset + size >= wvparse->upstream_length) {
+ GST_DEBUG_OBJECT (wvparse, "EOS: %" G_GINT64_FORMAT " + %u > %"
+ G_GINT64_FORMAT, offset, size, wvparse->upstream_length);
+ flow_ret = GST_FLOW_UNEXPECTED;
+ goto done;
+ }
}
- gst_wavpack_read_header (header, GST_BUFFER_DATA (buf));
- gst_buffer_unref (buf);
+ flow_ret = gst_pad_pull_range (wvparse->sinkpad, offset, size, &buf);
- if (GST_FLOW_OK != gst_pad_pull_range (wavpackparse->sinkpad,
- wavpackparse->flushed_bytes, header->ckSize + 8, &buf)) {
- wavpackparse->eos = TRUE;
- gst_pad_push_event (wavpackparse->srcpad, gst_event_new_eos ());
- goto done;
+ if (flow_ret != GST_FLOW_OK) {
+ GST_DEBUG_OBJECT (wvparse, "pull_range (%" G_GINT64_FORMAT ", %u) "
+ "failed, flow: %s", offset, size, gst_flow_get_name (flow_ret));
+ return NULL;
}
- num = GST_BUFFER_SIZE (buf);
- if (num != header->ckSize + 8) {
+ if (GST_BUFFER_SIZE (buf) < size) {
+ GST_DEBUG_OBJECT (wvparse, "Short read at offset %" G_GINT64_FORMAT
+ ", got only %u of %u bytes", offset, GST_BUFFER_SIZE (buf), size);
gst_buffer_unref (buf);
- wavpackparse->eos = TRUE;
- gst_pad_push_event (wavpackparse->srcpad, gst_event_new_eos ());
- goto done;
+ buf = NULL;
+ flow_ret = GST_FLOW_UNEXPECTED;
}
- if (!GST_IS_PAD (wavpackparse->srcpad)) {
+done:
+ if (flow)
+ *flow = flow_ret;
+ return buf;
+}
+
+static gboolean
+gst_wavpack_parse_create_src_pad (GstWavpackParse * wvparse, GstBuffer * buf,
+ WavpackHeader * header)
+{
+ WavpackMetadata meta;
+ GstCaps *caps = NULL;
+ guchar *bufptr;
+
+ g_assert (wvparse->srcpad == NULL);
- guchar *bufptr = GST_BUFFER_DATA (buf) + sizeof (WavpackHeader);
- GstCaps *caps = NULL;
- WavpackMetadata meta;
+ bufptr = GST_BUFFER_DATA (buf) + sizeof (WavpackHeader);
- while (read_metadata_buff (&meta, GST_BUFFER_DATA (buf), &bufptr)) {
- if (meta.id == ID_WVC_BITSTREAM) {
+ while (read_metadata_buff (&meta, GST_BUFFER_DATA (buf), &bufptr)) {
+ switch (meta.id) {
+ case ID_WVC_BITSTREAM:{
caps = gst_caps_new_simple ("audio/x-wavpack-correction",
"framed", G_TYPE_BOOLEAN, TRUE, NULL);
- if (GST_IS_PAD (wavpackparse->srcpad)) {
- gst_object_unref (wavpackparse->srcpad);
- }
- wavpackparse->srcpad =
+ wvparse->srcpad =
gst_pad_new_from_template (gst_element_class_get_pad_template
- (GST_ELEMENT_GET_CLASS (wavpackparse), "wvcsrc"), "wvcsrc");
- } else if (meta.id == ID_RIFF_HEADER) {
- WaveHeader *wheader = g_malloc (sizeof (WaveHeader));
+ (GST_ELEMENT_GET_CLASS (wvparse), "wvcsrc"), "wvcsrc");
+ break;
+ }
+ case ID_RIFF_HEADER:{
+ WaveHeader wheader;
/* skip RiffChunkHeader and ChunkHeader */
- g_memmove (wheader, meta.data + 20, sizeof (WaveHeader));
- little_endian_to_native (wheader, WaveHeaderFormat);
- wavpackparse->samplerate = wheader->SampleRate;
- wavpackparse->channels = wheader->NumChannels;
- wavpackparse->total_samples = header->total_samples;
+ g_memmove (&wheader, meta.data + 20, sizeof (WaveHeader));
+ little_endian_to_native (&wheader, WaveHeaderFormat);
+ wvparse->samplerate = wheader.SampleRate;
+ wvparse->channels = wheader.NumChannels;
+ wvparse->total_samples = header->total_samples;
caps = gst_caps_new_simple ("audio/x-wavpack",
- "width", G_TYPE_INT, wheader->BitsPerSample,
- "channels", G_TYPE_INT, wavpackparse->channels,
- "rate", G_TYPE_INT, wavpackparse->samplerate,
+ "width", G_TYPE_INT, wheader.BitsPerSample,
+ "channels", G_TYPE_INT, wvparse->channels,
+ "rate", G_TYPE_INT, wvparse->samplerate,
"framed", G_TYPE_BOOLEAN, TRUE, NULL);
- if (GST_IS_PAD (wavpackparse->srcpad)) {
- gst_object_unref (wavpackparse->srcpad);
- }
- wavpackparse->srcpad =
+ wvparse->srcpad =
gst_pad_new_from_template (gst_element_class_get_pad_template
- (GST_ELEMENT_GET_CLASS (wavpackparse), "src"), "src");
+ (GST_ELEMENT_GET_CLASS (wvparse), "src"), "src");
+ break;
+ }
+ default:{
+ GST_WARNING_OBJECT (wvparse, "unhandled ID: 0x%02x", meta.id);
+ break;
}
}
+ if (caps != NULL)
+ break;
+ }
- if (!(caps && GST_IS_PAD (wavpackparse->srcpad))) {
- gst_buffer_unref (buf);
- goto done;
- }
+ if (caps == NULL || wvparse->srcpad == NULL)
+ return FALSE;
- gst_pad_set_query_function (wavpackparse->srcpad,
- gst_wavpack_parse_src_query);
- gst_pad_set_event_function (wavpackparse->srcpad,
- gst_wavpack_parse_src_event);
+ GST_DEBUG_OBJECT (wvparse, "Added src pad with caps %" GST_PTR_FORMAT, caps);
- gst_pad_set_caps (wavpackparse->srcpad, caps);
- gst_pad_use_fixed_caps (wavpackparse->srcpad);
+ gst_pad_set_query_function (wvparse->srcpad,
+ GST_DEBUG_FUNCPTR (gst_wavpack_parse_src_query));
+ gst_pad_set_event_function (wvparse->srcpad,
+ GST_DEBUG_FUNCPTR (gst_wavpack_parse_src_event));
- gst_element_add_pad (GST_ELEMENT (wavpackparse), wavpackparse->srcpad);
+ gst_pad_set_caps (wvparse->srcpad, caps);
+ gst_pad_use_fixed_caps (wvparse->srcpad);
+ gst_object_ref (wvparse->srcpad);
+ gst_element_add_pad (GST_ELEMENT (wvparse), wvparse->srcpad);
+ gst_element_no_more_pads (GST_ELEMENT (wvparse));
+
+ return TRUE;
+}
+
+static void
+gst_wavpack_parse_loop (GstElement * element)
+{
+ GstWavpackParse *wavpackparse = GST_WAVPACK_PARSE (element);
+ GstFlowReturn flow_ret;
+ WavpackHeader header = { {0,}, 0, };
+ GstBuffer *buf = NULL;
+
+ GST_LOG_OBJECT (wavpackparse, "Current offset: %" G_GINT64_FORMAT,
+ wavpackparse->current_offset);
+
+ buf = gst_wavpack_parse_pull_buffer (wavpackparse,
+ wavpackparse->current_offset, sizeof (WavpackHeader), &flow_ret);
+
+ if (buf == NULL && flow_ret == GST_FLOW_UNEXPECTED) {
+ goto eos;
+ } else if (buf == NULL) {
+ goto pause;
}
- if (wavpackparse->need_discont) {
- if (GST_IS_PAD (wavpackparse->srcpad)) {
- gst_pad_push_event (wavpackparse->srcpad,
- gst_event_new_new_segment (FALSE, 1.0, GST_FORMAT_TIME,
- wavpackparse->timestamp, GST_CLOCK_TIME_NONE, 0));
+ gst_wavpack_read_header (&header, GST_BUFFER_DATA (buf));
+ gst_buffer_unref (buf);
+
+ GST_LOG_OBJECT (wavpackparse, "Read header at offset %" G_GINT64_FORMAT
+ ": chunk size = %u+8", wavpackparse->current_offset, header.ckSize);
+
+ buf = gst_wavpack_parse_pull_buffer (wavpackparse,
+ wavpackparse->current_offset, header.ckSize + 8, &flow_ret);
- wavpackparse->need_discont = FALSE;
+ if (buf == NULL && flow_ret == GST_FLOW_UNEXPECTED) {
+ goto eos;
+ } else if (buf == NULL) {
+ goto pause;
+ }
+
+ if (wavpackparse->srcpad == NULL) {
+ if (!gst_wavpack_parse_create_src_pad (wavpackparse, buf, &header)) {
+ GST_ELEMENT_ERROR (wavpackparse, STREAM, DECODE, (NULL), (NULL));
+ goto pause;
}
}
- wavpackparse->flushed_bytes += header->ckSize + 8;
+ 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;
- wavpackparse->timestamp =
- ((gdouble) header->block_index / (gdouble) wavpackparse->samplerate) *
- GST_SECOND;
- GST_BUFFER_TIMESTAMP (buf) = wavpackparse->timestamp;
- GST_BUFFER_DURATION (buf) =
- ((gdouble) header->block_samples / (gdouble) wavpackparse->samplerate) *
- GST_SECOND;
+ if (wavpackparse->need_newsegment) {
+ if (gst_wavpack_parse_send_newsegment (wavpackparse, FALSE))
+ wavpackparse->need_newsegment = FALSE;
+ }
+
+ 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));
- if (GST_FLOW_OK != gst_pad_push (wavpackparse->srcpad, buf)) {
- gst_buffer_unref (buf);
+ 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);
+ if (flow_ret != GST_FLOW_OK) {
+ GST_DEBUG_OBJECT (wavpackparse, "Push failed, flow: %s",
+ gst_flow_get_name (flow_ret));
+ goto pause;
}
-done:
- GST_PAD_STREAM_UNLOCK (wavpackparse->sinkpad);
return;
+eos:
+ {
+ GST_DEBUG_OBJECT (wavpackparse, "sending EOS");
+ if (wavpackparse->srcpad) {
+ gst_pad_push_event (wavpackparse->srcpad, gst_event_new_eos ());
+ }
+ /* fall through and pause task */
+ }
+pause:
+ {
+ GST_DEBUG_OBJECT (wavpackparse, "Pausing task");
+ gst_pad_pause_task (wavpackparse->sinkpad);
+ return;
+ }
}
static GstStateChangeReturn
gst_wavpack_parse_change_state (GstElement * element, GstStateChange transition)
{
- GstWavpackParse *wavpackparse = GST_WAVPACK_PARSE (element);
+ GstWavpackParse *wvparse = GST_WAVPACK_PARSE (element);
GstStateChangeReturn ret = GST_STATE_CHANGE_SUCCESS;
switch (transition) {
case GST_STATE_CHANGE_READY_TO_PAUSED:
- wavpackparse->flushed_bytes = 0;
- wavpackparse->need_discont = TRUE;
- wavpackparse->eos = FALSE;
- break;
- case GST_STATE_CHANGE_PAUSED_TO_PLAYING:
- {
- GstQuery *query;
- GstFormat format = GST_FORMAT_BYTES;
-
- query = gst_query_new_duration (GST_FORMAT_BYTES);
- if (gst_pad_query (GST_PAD_PEER (wavpackparse->sinkpad), query)) {
-
- gst_query_parse_duration (query, &format,
- (gint64 *) & wavpackparse->duration);
-
- if (format != GST_FORMAT_BYTES) {
- wavpackparse->duration = -1;
- ret = GST_STATE_CHANGE_FAILURE;
- }
-
- } else {
- wavpackparse->duration = -1;
- ret = GST_STATE_CHANGE_FAILURE;
- }
- gst_query_unref (query);
- }
- break;
+ gst_segment_init (&wvparse->segment, GST_FORMAT_DEFAULT);
+ wvparse->segment.last_stop = 0;
default:
break;
}
- if (GST_ELEMENT_CLASS (parent)->change_state)
- ret = GST_ELEMENT_CLASS (parent)->change_state (element, transition);
+ if (GST_ELEMENT_CLASS (parent_class)->change_state)
+ ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
switch (transition) {
- case GST_STATE_CHANGE_PLAYING_TO_PAUSED:
- break;
case GST_STATE_CHANGE_PAUSED_TO_READY:
- wavpackparse->seek_pending = FALSE;
+ gst_wavpack_parse_reset (wvparse);
break;
default:
break;
@@ -658,7 +814,6 @@ gst_wavpack_parse_change_state (GstElement * element, GstStateChange transition)
static gboolean
gst_wavepack_parse_sink_activate (GstPad * sinkpad)
{
-
if (gst_pad_check_pull_range (sinkpad)) {
return gst_pad_activate_pull (sinkpad, TRUE);
} else {
@@ -669,11 +824,9 @@ gst_wavepack_parse_sink_activate (GstPad * sinkpad)
static gboolean
gst_wavepack_parse_sink_activate_pull (GstPad * sinkpad, gboolean active)
{
-
gboolean result;
if (active) {
-
result = gst_pad_start_task (sinkpad,
(GstTaskFunction) gst_wavpack_parse_loop, GST_PAD_PARENT (sinkpad));
} else {
@@ -681,14 +834,11 @@ gst_wavepack_parse_sink_activate_pull (GstPad * sinkpad, gboolean active)
}
return result;
-
- return TRUE;
}
gboolean
gst_wavpack_parse_plugin_init (GstPlugin * plugin)
{
-
if (!gst_element_register (plugin, "wavpackparse",
GST_RANK_PRIMARY, GST_TYPE_WAVPACK_PARSE)) {
return FALSE;