summaryrefslogtreecommitdiffstats
path: root/ext/ivorbis
diff options
context:
space:
mode:
Diffstat (limited to 'ext/ivorbis')
-rw-r--r--ext/ivorbis/vorbis.c3
-rw-r--r--ext/ivorbis/vorbisfile.c515
2 files changed, 335 insertions, 183 deletions
diff --git a/ext/ivorbis/vorbis.c b/ext/ivorbis/vorbis.c
index 3394a77a..592d9d2a 100644
--- a/ext/ivorbis/vorbis.c
+++ b/ext/ivorbis/vorbis.c
@@ -30,9 +30,6 @@ extern GType ivorbisfile_get_type (void);
static gboolean
plugin_init (GstPlugin * plugin)
{
- if (!gst_library_load ("gstbytestream"))
- return FALSE;
-
if (!gst_element_register (plugin, "tremor", GST_RANK_PRIMARY,
ivorbisfile_get_type ()))
return FALSE;
diff --git a/ext/ivorbis/vorbisfile.c b/ext/ivorbis/vorbisfile.c
index db2b601b..6ab26714 100644
--- a/ext/ivorbis/vorbisfile.c
+++ b/ext/ivorbis/vorbisfile.c
@@ -24,20 +24,20 @@
#include <gst/gst.h>
#include <tremor/ivorbiscodec.h>
#include <tremor/ivorbisfile.h>
-#include <gst/bytestream/bytestream.h>
+#include <gst/base/gstadapter.h>
GST_DEBUG_CATEGORY_STATIC (ivorbisfile_debug);
#define GST_CAT_DEFAULT ivorbisfile_debug
-#define GST_TYPE_IVORBISFILE \
+#define GST_TYPE_IVORBISFILE \
(ivorbisfile_get_type())
-#define GST_IVORBISFILE(obj) \
+#define GST_IVORBISFILE(obj) \
(G_TYPE_CHECK_INSTANCE_CAST((obj),GST_TYPE_IVORBISFILE,Ivorbisfile))
-#define GST_IVORBISFILE_CLASS(klass) \
+#define GST_IVORBISFILE_CLASS(klass) \
(G_TYPE_CHECK_CLASS_CAST((klass),GST_TYPE_IVORBISFILE,IvorbisfileClass))
-#define GST_IS_IVORBISFILE(obj) \
+#define GST_IS_IVORBISFILE(obj) \
(G_TYPE_CHECK_INSTANCE_TYPE((obj),GST_TYPE_IVORBISFILE))
-#define GST_IS_IVORBISFILE_CLASS(obj) \
+#define GST_IS_IVORBISFILE_CLASS(obj) \
(G_TYPE_CHECK_CLASS_TYPE((klass),GST_TYPE_IVORBISFILE))
typedef struct _Ivorbisfile Ivorbisfile;
@@ -48,7 +48,8 @@ struct _Ivorbisfile
GstElement element;
GstPad *sinkpad, *srcpad;
- GstByteStream *bs;
+ GstAdapter *adapter;
+ guint64 adapterOffset;
OggVorbis_File vf;
gint current_link;
@@ -65,6 +66,10 @@ struct _Ivorbisfile
guint64 total_bytes;
guint64 offset;
+ gint rate;
+ gint channels;
+ gint width;
+
GstCaps *metadata;
GstCaps *streaminfo;
};
@@ -109,30 +114,37 @@ static void gst_ivorbisfile_init (Ivorbisfile * ivorbisfile);
static GstStateChangeReturn
gst_ivorbisfile_change_state (GstElement * element, GstStateChange transition);
-static const GstFormat *gst_ivorbisfile_get_formats (GstPad * pad);
static gboolean gst_ivorbisfile_src_convert (GstPad * pad,
GstFormat src_format,
gint64 src_value, GstFormat * dest_format, gint64 * dest_value);
static gboolean gst_ivorbisfile_sink_convert (GstPad * pad,
GstFormat src_format,
gint64 src_value, GstFormat * dest_format, gint64 * dest_value);
-static const GstQueryType *gst_ivorbisfile_get_query_types (GstPad * pad);
+static const GstQueryType *gst_ivorbisfile_get_src_query_types (GstPad * pad);
+
+static gboolean gst_ivorbisfile_src_query (GstPad * pad, GstQuery * query);
-static gboolean gst_ivorbisfile_src_query (GstPad * pad,
- GstQueryType type, GstFormat * format, gint64 * value);
-static const GstEventMask *gst_ivorbisfile_get_event_masks (GstPad * pad);
static gboolean gst_ivorbisfile_src_event (GstPad * pad, GstEvent * event);
+static gboolean gst_ivorbisfile_sink_event (GstPad * pad, GstEvent * event);
+
+static const GstQueryType *gst_ivorbisfile_get_sink_query_types (GstPad * pad);
+
+static gboolean gst_ivorbisfile_sink_query (GstPad * pad, GstQuery * query);
+
static void gst_ivorbisfile_get_property (GObject * object,
guint prop_id, GValue * value, GParamSpec * pspec);
static void gst_ivorbisfile_set_property (GObject * object,
guint prop_id, const GValue * value, GParamSpec * pspec);
-static void gst_ivorbisfile_loop (GstElement * element);
+static gboolean gst_ivorbisfile_sink_activate (GstPad * sinkpad);
-static GstElementClass *parent_class = NULL;
+static gboolean
+gst_ivorbisfile_sink_activate_pull (GstPad * sinkpad, gboolean active);
+
+static void gst_ivorbisfile_loop (GstPad * pad);
-//static guint gst_ivorbisfile_signals[LAST_SIGNAL] = { 0 };
+static GstElementClass *parent_class = NULL;
static GstFormat logical_stream_format;
@@ -230,6 +242,9 @@ gst_ivorbisfile_class_init (IvorbisfileClass * klass)
parent_class = g_type_class_ref (GST_TYPE_ELEMENT);
+ gobject_class->get_property = gst_ivorbisfile_get_property;
+ gobject_class->set_property = gst_ivorbisfile_set_property;
+
g_object_class_install_property (gobject_class, ARG_METADATA,
g_param_spec_boxed ("metadata", "Metadata", "(logical) Stream metadata",
GST_TYPE_CAPS, G_PARAM_READABLE));
@@ -237,8 +252,6 @@ gst_ivorbisfile_class_init (IvorbisfileClass * klass)
g_param_spec_boxed ("streaminfo", "stream",
"(logical) Stream information", GST_TYPE_CAPS, G_PARAM_READABLE));
- gobject_class->get_property = gst_ivorbisfile_get_property;
- gobject_class->set_property = gst_ivorbisfile_set_property;
gstelement_class->change_state = gst_ivorbisfile_change_state;
}
@@ -249,26 +262,27 @@ gst_ivorbisfile_init (Ivorbisfile * ivorbisfile)
ivorbisfile->sinkpad = gst_pad_new_from_template (gst_vorbisdec_sink_template,
"sink");
gst_element_add_pad (GST_ELEMENT (ivorbisfile), ivorbisfile->sinkpad);
- gst_pad_set_formats_function (ivorbisfile->sinkpad,
- gst_ivorbisfile_get_formats);
- gst_pad_set_convert_function (ivorbisfile->sinkpad,
- gst_ivorbisfile_sink_convert);
- gst_element_set_loop_function (GST_ELEMENT (ivorbisfile),
- gst_ivorbisfile_loop);
+ gst_pad_set_query_type_function (ivorbisfile->sinkpad,
+ gst_ivorbisfile_get_sink_query_types);
+ gst_pad_set_query_function (ivorbisfile->sinkpad, gst_ivorbisfile_sink_query);
+
+ gst_pad_set_activate_function (ivorbisfile->sinkpad,
+ gst_ivorbisfile_sink_activate);
+ gst_pad_set_activatepull_function (ivorbisfile->sinkpad,
+ gst_ivorbisfile_sink_activate_pull);
+
+ gst_pad_set_event_function (ivorbisfile->sinkpad, gst_ivorbisfile_sink_event);
+
ivorbisfile->srcpad =
gst_pad_new_from_template (gst_vorbisdec_src_template, "src");
gst_element_add_pad (GST_ELEMENT (ivorbisfile), ivorbisfile->srcpad);
- gst_pad_set_formats_function (ivorbisfile->srcpad,
- gst_ivorbisfile_get_formats);
+
gst_pad_set_query_type_function (ivorbisfile->srcpad,
- gst_ivorbisfile_get_query_types);
+ gst_ivorbisfile_get_src_query_types);
gst_pad_set_query_function (ivorbisfile->srcpad, gst_ivorbisfile_src_query);
- gst_pad_set_event_mask_function (ivorbisfile->srcpad,
- gst_ivorbisfile_get_event_masks);
+
gst_pad_set_event_function (ivorbisfile->srcpad, gst_ivorbisfile_src_event);
- gst_pad_set_convert_function (ivorbisfile->srcpad,
- gst_ivorbisfile_src_convert);
ivorbisfile->total_bytes = 0;
ivorbisfile->offset = 0;
@@ -277,6 +291,11 @@ gst_ivorbisfile_init (Ivorbisfile * ivorbisfile)
ivorbisfile->metadata = NULL;
ivorbisfile->streaminfo = NULL;
ivorbisfile->current_link = -1;
+
+ ivorbisfile->rate = -1;
+ ivorbisfile->channels = -1;
+ ivorbisfile->width = -1;
+
}
/* the next four functions are the ov callbacks we provide to ivorbisfile
@@ -285,15 +304,15 @@ gst_ivorbisfile_init (Ivorbisfile * ivorbisfile)
static size_t
gst_ivorbisfile_read (void *ptr, size_t size, size_t nmemb, void *datasource)
{
- guint32 got_bytes = 0;
- guint8 *data;
size_t read_size = size * nmemb;
-
+ GstBuffer *buf = NULL;
Ivorbisfile *ivorbisfile = GST_IVORBISFILE (datasource);
+ size_t ret;
GST_DEBUG ("read %d", read_size);
/* make sure we don't go to EOS */
+
if (!ivorbisfile->may_eos && ivorbisfile->total_bytes &&
ivorbisfile->offset + read_size > ivorbisfile->total_bytes) {
read_size = ivorbisfile->total_bytes - ivorbisfile->offset;
@@ -302,50 +321,25 @@ gst_ivorbisfile_read (void *ptr, size_t size, size_t nmemb, void *datasource)
if (read_size == 0 || ivorbisfile->eos)
return 0;
- while (got_bytes == 0) {
- got_bytes = gst_bytestream_peek_bytes (ivorbisfile->bs, &data, read_size);
- if (got_bytes < read_size) {
- GstEvent *event;
- guint32 avail;
-
- gst_bytestream_get_status (ivorbisfile->bs, &avail, &event);
-
- switch (GST_EVENT_TYPE (event)) {
- case GST_EVENT_EOS:
- GST_DEBUG ("eos");
- ivorbisfile->eos = TRUE;
- if (avail == 0) {
- gst_event_unref (event);
- return 0;
- }
- break;
- case GST_EVENT_DISCONTINUOUS:
- GST_DEBUG ("discont");
- ivorbisfile->need_discont = TRUE;
- default:
- break;
- }
- gst_event_unref (event);
- if (avail > 0)
- got_bytes = gst_bytestream_peek_bytes (ivorbisfile->bs, &data, avail);
- else
- got_bytes = 0;
- }
+ if (GST_FLOW_OK != gst_pad_pull_range (ivorbisfile->sinkpad,
+ ivorbisfile->offset, read_size, &buf)) {
+ return 0;
}
- memcpy (ptr, data, got_bytes);
- gst_bytestream_flush_fast (ivorbisfile->bs, got_bytes);
+ memcpy (ptr, GST_BUFFER_DATA (buf), GST_BUFFER_SIZE (buf));
- ivorbisfile->offset += got_bytes;
+ ivorbisfile->offset += GST_BUFFER_SIZE (buf);
- return got_bytes / size;
+ ret = GST_BUFFER_SIZE (buf) / size;
+ gst_buffer_unref (buf);
+
+ return ret;
}
static int
gst_ivorbisfile_seek (void *datasource, int64_t offset, int whence)
{
Ivorbisfile *ivorbisfile = GST_IVORBISFILE (datasource);
- GstSeekType method;
guint64 pending_offset = ivorbisfile->offset;
gboolean need_total = FALSE;
@@ -357,24 +351,22 @@ gst_ivorbisfile_seek (void *datasource, int64_t offset, int whence)
GST_DEBUG ("seek %" G_GINT64_FORMAT " %d", offset, whence);
if (whence == SEEK_SET) {
- method = GST_SEEK_METHOD_SET;
pending_offset = offset;
+ ivorbisfile->adapterOffset = offset;
} else if (whence == SEEK_CUR) {
- method = GST_SEEK_METHOD_CUR;
pending_offset += offset;
+ ivorbisfile->adapterOffset += offset;
} else if (whence == SEEK_END) {
- method = GST_SEEK_METHOD_END;
need_total = TRUE;
pending_offset = ivorbisfile->total_bytes - offset;
+ ivorbisfile->adapterOffset = ivorbisfile->total_bytes - offset;
} else
return -1;
- if (!gst_bytestream_seek (ivorbisfile->bs, offset, method))
- return -1;
ivorbisfile->offset = pending_offset;
if (need_total)
- ivorbisfile->total_bytes = gst_bytestream_tell (ivorbisfile->bs) + offset;
+ ivorbisfile->total_bytes = ivorbisfile->adapterOffset + offset;
return 0;
}
@@ -392,7 +384,7 @@ gst_ivorbisfile_tell (void *datasource)
Ivorbisfile *ivorbisfile = GST_IVORBISFILE (datasource);
long result;
- result = gst_bytestream_tell (ivorbisfile->bs);
+ result = ivorbisfile->adapterOffset;
GST_DEBUG ("tell %ld", result);
@@ -513,40 +505,89 @@ gst_ivorbisfile_new_link (Ivorbisfile * ivorbisfile, gint link)
"depth", G_TYPE_INT, 16,
"rate", G_TYPE_INT, vi->rate, "channels", G_TYPE_INT, vi->channels, NULL);
- if (gst_pad_try_set_caps (ivorbisfile->srcpad, caps) <= 0) {
+ ivorbisfile->rate = vi->rate;
+ ivorbisfile->channels = vi->channels;
+ ivorbisfile->width = 16;
+
+ if (gst_pad_set_caps (ivorbisfile->srcpad, caps) <= 0) {
res = FALSE;
}
- gst_caps_free (caps);
+ gst_caps_unref (caps);
return TRUE;
}
+
+static gboolean
+gst_ivorbisfile_sink_activate (GstPad * sinkpad)
+{
+
+ Ivorbisfile *ivorbisfile;
+
+ ivorbisfile = GST_IVORBISFILE (GST_PAD_PARENT (sinkpad));
+
+ if (gst_pad_check_pull_range (sinkpad)) {
+ /* FIX ME */
+ /* ivorbisfile->vf.seekable = TRUE; */
+ ivorbisfile->vf.seekable = FALSE;
+ return gst_pad_activate_pull (sinkpad, TRUE);
+ }
+
+ ivorbisfile->vf.seekable = FALSE;
+ return FALSE;
+
+
+}
+
+
+static gboolean
+gst_ivorbisfile_sink_activate_pull (GstPad * sinkpad, gboolean active)
+{
+
+ gboolean result;
+
+ if (active) {
+ /* if we have a scheduler we can start the task */
+ result = gst_pad_start_task (sinkpad,
+ (GstTaskFunction) gst_ivorbisfile_loop, sinkpad);
+ } else {
+ result = gst_pad_stop_task (sinkpad);
+ }
+
+ return result;
+}
+
+
+
static void
-gst_ivorbisfile_loop (GstElement * element)
+gst_ivorbisfile_loop (GstPad * pad)
{
- Ivorbisfile *ivorbisfile = GST_IVORBISFILE (element);
+ Ivorbisfile *ivorbisfile = GST_IVORBISFILE (GST_PAD_PARENT (pad));
GstBuffer *outbuf;
long ret;
GstClockTime time;
gint64 samples;
gint link;
+ if (ivorbisfile->eos)
+ return;
+
/* this function needs to go first since you don't want to be messing
* with an unset vf ;) */
if (ivorbisfile->restart) {
+
ivorbisfile->offset = 0;
ivorbisfile->total_bytes = 0;
ivorbisfile->may_eos = FALSE;
- ivorbisfile->vf.seekable = gst_bytestream_seek (ivorbisfile->bs, 0,
- GST_SEEK_METHOD_SET);
+ ivorbisfile->adapterOffset = 0;
GST_DEBUG ("ivorbisfile: seekable: %s\n",
ivorbisfile->vf.seekable ? "yes" : "no");
/* open our custom ivorbisfile data object with the callbacks we provide */
if (ov_open_callbacks (ivorbisfile, &ivorbisfile->vf, NULL, 0,
ivorbisfile_ov_callbacks) < 0) {
- GST_ELEMENT_ERROR (element, STREAM, DECODE, (NULL), (NULL));
+ GST_ELEMENT_ERROR (ivorbisfile, STREAM, DECODE, (NULL), (NULL));
return;
}
ivorbisfile->need_discont = TRUE;
@@ -615,24 +656,21 @@ gst_ivorbisfile_loop (GstElement * element)
GST_BUFFER_DATA (outbuf) = g_malloc (4096);
GST_BUFFER_SIZE (outbuf) = 4096;
+ ret = ov_read (&ivorbisfile->vf,
+ (char *) GST_BUFFER_DATA (outbuf), GST_BUFFER_SIZE (outbuf), &link);
+
/* get current time for discont and buffer timestamp */
time = (GstClockTime) (ov_time_tell (&ivorbisfile->vf) * GST_SECOND);
- ret = ov_read (&ivorbisfile->vf,
- GST_BUFFER_DATA (outbuf), GST_BUFFER_SIZE (outbuf), &link);
-
if (ret == 0) {
GST_DEBUG ("eos");
/* send EOS event */
- /*ov_clear (&ivorbisfile->vf); */
+
ivorbisfile->restart = TRUE;
gst_buffer_unref (outbuf);
/* if the pad is not usable, don't push it out */
- if (GST_PAD_IS_USABLE (ivorbisfile->srcpad)) {
- gst_pad_push (ivorbisfile->srcpad,
- GST_DATA (gst_event_new (GST_EVENT_EOS)));
- }
- gst_element_set_eos (element);
+ gst_pad_push_event (ivorbisfile->srcpad, gst_event_new_eos ());
+
return;
} else if (ret < 0) {
g_warning ("ivorbisfile: decoding error");
@@ -640,24 +678,30 @@ gst_ivorbisfile_loop (GstElement * element)
return;
} else {
if (ivorbisfile->need_discont) {
- GstEvent *discont;
+ GstEvent *event;
ivorbisfile->need_discont = FALSE;
- /* if the pad is not usable, don't push it out */
- if (GST_PAD_IS_USABLE (ivorbisfile->srcpad)) {
- /* get stream stats */
- samples = (gint64) (ov_pcm_tell (&ivorbisfile->vf));
+ /* get stream stats */
+ samples = (gint64) (ov_pcm_tell (&ivorbisfile->vf));
- discont = gst_event_new_discontinuous (FALSE, GST_FORMAT_TIME, time,
- GST_FORMAT_DEFAULT, samples, NULL);
+ event =
+ gst_event_new_newsegment (FALSE, 1.0, GST_FORMAT_TIME, time,
+ GST_CLOCK_TIME_NONE, 0);
+
+ gst_pad_push_event (ivorbisfile->srcpad, event);
- gst_pad_push (ivorbisfile->srcpad, GST_DATA (discont));
- }
}
+ gst_buffer_set_caps (outbuf, GST_PAD_CAPS (ivorbisfile->srcpad));
+
GST_BUFFER_SIZE (outbuf) = ret;
- GST_BUFFER_TIMESTAMP (outbuf) = time;
+ /* FIX ME TO SET RIGHT TIMESTAMP
+ gint bufsize = ret / (ivorbisfile->width / 8);
+ GST_BUFFER_TIMESTAMP (outbuf) = time;
+ GST_BUFFER_DURATION (outbuf) = GST_SECOND * bufsize / (ivorbisfile->rate * ivorbisfile->channels);
+ */
+
ivorbisfile->may_eos = TRUE;
@@ -665,34 +709,10 @@ gst_ivorbisfile_loop (GstElement * element)
ivorbisfile->total_bytes += GST_BUFFER_SIZE (outbuf);
}
- if (GST_PAD_IS_USABLE (ivorbisfile->srcpad))
- gst_pad_push (ivorbisfile->srcpad, GST_DATA (outbuf));
- else
- gst_buffer_unref (outbuf);
- }
-}
-
-static const GstFormat *
-gst_ivorbisfile_get_formats (GstPad * pad)
-{
- static GstFormat src_formats[] = {
- GST_FORMAT_TIME,
- GST_FORMAT_BYTES,
- GST_FORMAT_DEFAULT,
- 0,
- 0
- };
- static GstFormat sink_formats[] = {
- GST_FORMAT_TIME,
- GST_FORMAT_BYTES,
- 0,
- 0
- };
+ gst_pad_push (ivorbisfile->srcpad, outbuf);
- src_formats[3] = logical_stream_format;
- sink_formats[2] = logical_stream_format;
+ }
- return (GST_PAD_IS_SRC (pad) ? src_formats : sink_formats);
}
static gboolean
@@ -706,7 +726,7 @@ gst_ivorbisfile_src_convert (GstPad * pad,
Ivorbisfile *ivorbisfile;
vorbis_info *vi;
- ivorbisfile = GST_IVORBISFILE (gst_pad_get_parent (pad));
+ ivorbisfile = GST_IVORBISFILE (GST_PAD_PARENT (pad));
vi = ov_info (&ivorbisfile->vf, -1);
bytes_per_sample = vi->channels * 2;
@@ -804,6 +824,32 @@ gst_ivorbisfile_src_convert (GstPad * pad,
}
static gboolean
+gst_ivorbisfile_sink_query (GstPad * pad, GstQuery * query)
+{
+ gboolean res;
+
+ switch (GST_QUERY_TYPE (query)) {
+ case GST_QUERY_CONVERT:
+ {
+ GstFormat src_fmt, dest_fmt;
+ gint64 src_val, dest_val;
+
+ /* peel off input */
+ gst_query_parse_convert (query, &src_fmt, &src_val, &dest_fmt, &dest_val);
+ if ((res = gst_ivorbisfile_sink_convert (pad, src_fmt, src_val,
+ &dest_fmt, &dest_val))) {
+ gst_query_set_convert (query, src_fmt, src_val, dest_fmt, dest_val);
+ }
+ break;
+ }
+ default:
+ res = FALSE;
+ break;
+ }
+ return res;
+}
+
+static gboolean
gst_ivorbisfile_sink_convert (GstPad * pad,
GstFormat src_format, gint64 src_value,
GstFormat * dest_format, gint64 * dest_value)
@@ -811,7 +857,7 @@ gst_ivorbisfile_sink_convert (GstPad * pad,
gboolean res = TRUE;
Ivorbisfile *ivorbisfile;
- ivorbisfile = GST_IVORBISFILE (gst_pad_get_parent (pad));
+ ivorbisfile = GST_IVORBISFILE (GST_PAD_PARENT (pad));
switch (src_format) {
case GST_FORMAT_BYTES:
@@ -851,11 +897,23 @@ gst_ivorbisfile_sink_convert (GstPad * pad,
}
static const GstQueryType *
-gst_ivorbisfile_get_query_types (GstPad * pad)
+gst_ivorbisfile_get_sink_query_types (GstPad * pad)
+{
+ static const GstQueryType types[] = {
+ GST_QUERY_CONVERT,
+ 0
+ };
+
+ return types;
+}
+
+static const GstQueryType *
+gst_ivorbisfile_get_src_query_types (GstPad * pad)
{
static const GstQueryType types[] = {
- GST_QUERY_TOTAL,
GST_QUERY_POSITION,
+ GST_QUERY_DURATION,
+ GST_QUERY_CONVERT,
0
};
@@ -864,103 +922,194 @@ gst_ivorbisfile_get_query_types (GstPad * pad)
/* handles queries for location in the stream in the requested format */
static gboolean
-gst_ivorbisfile_src_query (GstPad * pad, GstQueryType type,
- GstFormat * format, gint64 * value)
+gst_ivorbisfile_src_query (GstPad * pad, GstQuery * query)
{
gboolean res = TRUE;
Ivorbisfile *ivorbisfile;
vorbis_info *vi;
- ivorbisfile = GST_IVORBISFILE (gst_pad_get_parent (pad));
+ ivorbisfile = GST_IVORBISFILE (GST_PAD_PARENT (pad));
vi = ov_info (&ivorbisfile->vf, -1);
- switch (type) {
- case GST_QUERY_TOTAL:
+ switch (GST_QUERY_TYPE (query)) {
+ case GST_QUERY_POSITION:
{
- switch (*format) {
+
+ GstFormat format;
+ GstFormat rformat = GST_FORMAT_TIME;
+ gint64 cur;
+ GstPad *peer;
+
+ /* save requested format */
+ gst_query_parse_position (query, &format, NULL);
+
+ /* query peer for current position in time */
+ gst_query_set_position (query, GST_FORMAT_TIME, -1);
+
+ if ((peer = gst_pad_get_peer (ivorbisfile->sinkpad)) == NULL)
+ goto error;
+
+ if (!gst_pad_query_position (peer, &rformat, &cur)) {
+ GST_LOG_OBJECT (ivorbisfile, "query on peer pad failed");
+ gst_object_unref (peer);
+ goto error;
+ }
+ gst_object_unref (peer);
+
+ if (format != rformat) {
+ gst_ivorbisfile_src_convert (pad, rformat, cur, &format, &cur);
+ }
+
+ switch (format) {
case GST_FORMAT_DEFAULT:
if (ivorbisfile->vf.seekable)
- *value = ov_pcm_total (&ivorbisfile->vf, -1);
+ cur = ov_pcm_tell (&ivorbisfile->vf);
else
- return FALSE;
+ cur = ivorbisfile->total_bytes / (vi->channels * 2);
break;
- case GST_FORMAT_BYTES:
+ case GST_FORMAT_TIME:
if (ivorbisfile->vf.seekable)
- *value = ov_pcm_total (&ivorbisfile->vf, -1) * vi->channels * 2;
+ cur = (gint64) (ov_time_tell (&ivorbisfile->vf) * GST_SECOND);
else
- return FALSE;
+ cur = ivorbisfile->total_bytes * GST_SECOND
+ / (vi->rate * vi->channels * 2);
break;
- case GST_FORMAT_TIME:
+ case GST_FORMAT_BYTES:
if (ivorbisfile->vf.seekable)
- *value =
- (gint64) (ov_time_total (&ivorbisfile->vf, -1) * GST_SECOND);
+ cur = ov_pcm_tell (&ivorbisfile->vf) * vi->channels * 2;
else
- return FALSE;
+ cur = ivorbisfile->total_bytes;
break;
default:
- if (*format == logical_stream_format) {
+ if (format == logical_stream_format) {
if (ivorbisfile->vf.seekable)
- *value = ivorbisfile->vf.links;
+ cur = ivorbisfile->current_link;
else
return FALSE;
} else
res = FALSE;
break;
}
+
+ gst_query_set_position (query, format, cur);
+
break;
}
- case GST_QUERY_POSITION:
- switch (*format) {
- case GST_FORMAT_TIME:
+ case GST_QUERY_DURATION:
+ {
+
+ GstFormat format;
+ GstFormat rformat = GST_FORMAT_TIME;
+ gint64 cur;
+ GstPad *peer;
+
+ /* save requested format */
+ gst_query_parse_position (query, &format, NULL);
+
+ /* query peer for current position in time */
+ gst_query_set_position (query, GST_FORMAT_TIME, -1);
+
+ if ((peer = gst_pad_get_peer (ivorbisfile->sinkpad)) == NULL)
+ goto error;
+
+ if (!gst_pad_query_position (peer, &rformat, &cur)) {
+ GST_LOG_OBJECT (ivorbisfile, "query on peer pad failed");
+ gst_object_unref (peer);
+ goto error;
+ }
+ gst_object_unref (peer);
+
+ if (format != rformat) {
+ gst_ivorbisfile_src_convert (pad, rformat, cur, &format, &cur);
+ }
+
+ switch (format) {
+ case GST_FORMAT_DEFAULT:
if (ivorbisfile->vf.seekable)
- *value = (gint64) (ov_time_tell (&ivorbisfile->vf) * GST_SECOND);
+ cur = ov_pcm_total (&ivorbisfile->vf, -1);
else
- *value = ivorbisfile->total_bytes * GST_SECOND
- / (vi->rate * vi->channels * 2);
+ return FALSE;
break;
case GST_FORMAT_BYTES:
if (ivorbisfile->vf.seekable)
- *value = ov_pcm_tell (&ivorbisfile->vf) * vi->channels * 2;
+ cur = ov_pcm_total (&ivorbisfile->vf, -1) * vi->channels * 2;
else
- *value = ivorbisfile->total_bytes;
+ return FALSE;
break;
- case GST_FORMAT_DEFAULT:
+ case GST_FORMAT_TIME:
if (ivorbisfile->vf.seekable)
- *value = ov_pcm_tell (&ivorbisfile->vf);
+ cur = (gint64) (ov_time_total (&ivorbisfile->vf, -1) * GST_SECOND);
else
- *value = ivorbisfile->total_bytes / (vi->channels * 2);
+ return FALSE;
break;
default:
- if (*format == logical_stream_format) {
+ if (format == logical_stream_format) {
if (ivorbisfile->vf.seekable)
- *value = ivorbisfile->current_link;
+ cur = ivorbisfile->vf.links;
else
return FALSE;
} else
res = FALSE;
break;
}
+
+ gst_query_set_position (query, format, cur);
+
+ break;
+ }
+ case GST_QUERY_CONVERT:
+ {
+ GstFormat src_fmt, dest_fmt;
+ gint64 src_val, dest_val;
+
+ /* peel off input */
+ gst_query_parse_convert (query, &src_fmt, &src_val, NULL, NULL);
+ if ((res = gst_ivorbisfile_src_convert (pad, src_fmt, src_val,
+ &dest_fmt, &dest_val))) {
+ gst_query_set_convert (query, src_fmt, src_val, dest_fmt, dest_val);
+ }
break;
+ }
default:
res = FALSE;
break;
}
return res;
+
+error:
+
+ return FALSE;
+
}
-static const GstEventMask *
-gst_ivorbisfile_get_event_masks (GstPad * pad)
+
+static gboolean
+gst_ivorbisfile_sink_event (GstPad * pad, GstEvent * event)
{
- static const GstEventMask masks[] = {
- {GST_EVENT_SEEK, GST_SEEK_METHOD_SET | GST_SEEK_FLAG_ACCURATE},
- {0,}
- };
- return masks;
+ Ivorbisfile *ivorbisfile;
+
+ ivorbisfile = GST_IVORBISFILE (GST_PAD_PARENT (pad));
+
+ switch (GST_EVENT_TYPE (event)) {
+ case GST_EVENT_EOS:
+ GST_DEBUG ("eos");
+ ov_clear (&ivorbisfile->vf);
+ ivorbisfile->eos = TRUE;
+ break;
+ case GST_EVENT_NEWSEGMENT:
+ GST_DEBUG ("discont");
+ ivorbisfile->need_discont = TRUE;
+ default:
+ break;
+ }
+
+ return gst_pad_event_default (pad, event);
}
+
/* handle events on src pad */
static gboolean
gst_ivorbisfile_src_event (GstPad * pad, GstEvent * event)
@@ -976,6 +1125,7 @@ gst_ivorbisfile_src_event (GstPad * pad, GstEvent * event)
gint64 offset;
vorbis_info *vi;
GstFormat format;
+ GstSeekFlags flags;
GST_DEBUG ("ivorbisfile: handling seek event on pad %s:%s",
GST_DEBUG_PAD_NAME (pad));
@@ -985,16 +1135,15 @@ gst_ivorbisfile_src_event (GstPad * pad, GstEvent * event)
return FALSE;
}
- offset = GST_EVENT_SEEK_OFFSET (event);
- format = GST_EVENT_SEEK_FORMAT (event);
+ gst_event_parse_seek (event, NULL, &format, &flags, NULL, &offset, NULL,
+ NULL);
switch (format) {
case GST_FORMAT_TIME:
ivorbisfile->seek_pending = TRUE;
ivorbisfile->seek_value = offset;
ivorbisfile->seek_format = format;
- ivorbisfile->seek_accurate = GST_EVENT_SEEK_FLAGS (event)
- & GST_SEEK_FLAG_ACCURATE;
+ ivorbisfile->seek_accurate = flags & GST_SEEK_FLAG_ACCURATE;
break;
case GST_FORMAT_BYTES:
vi = ov_info (&ivorbisfile->vf, -1);
@@ -1009,16 +1158,14 @@ gst_ivorbisfile_src_event (GstPad * pad, GstEvent * event)
ivorbisfile->seek_pending = TRUE;
ivorbisfile->seek_value = offset;
ivorbisfile->seek_format = format;
- ivorbisfile->seek_accurate = GST_EVENT_SEEK_FLAGS (event)
- & GST_SEEK_FLAG_ACCURATE;
+ ivorbisfile->seek_accurate = flags & GST_SEEK_FLAG_ACCURATE;
break;
default:
if (format == logical_stream_format) {
ivorbisfile->seek_pending = TRUE;
ivorbisfile->seek_value = offset;
ivorbisfile->seek_format = format;
- ivorbisfile->seek_accurate = GST_EVENT_SEEK_FLAGS (event)
- & GST_SEEK_FLAG_ACCURATE;
+ ivorbisfile->seek_accurate = flags & GST_SEEK_FLAG_ACCURATE;
} else {
GST_DEBUG ("unhandled seek format");
res = FALSE;
@@ -1040,31 +1187,39 @@ done:
static GstStateChangeReturn
gst_ivorbisfile_change_state (GstElement * element, GstStateChange transition)
{
+ GstStateChangeReturn ret = GST_STATE_CHANGE_SUCCESS;
Ivorbisfile *ivorbisfile = GST_IVORBISFILE (element);
switch (transition) {
case GST_STATE_CHANGE_NULL_TO_READY:
+ /* fall through */
case GST_STATE_CHANGE_READY_TO_PAUSED:
ivorbisfile->restart = TRUE;
- ivorbisfile->bs = gst_bytestream_new (ivorbisfile->sinkpad);
break;
case GST_STATE_CHANGE_PAUSED_TO_PLAYING:
ivorbisfile->eos = FALSE;
+ break;
+ default:
+ break;
+ }
+
+ 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:
ov_clear (&ivorbisfile->vf);
- gst_bytestream_destroy (ivorbisfile->bs);
break;
case GST_STATE_CHANGE_READY_TO_NULL:
default:
break;
}
- if (GST_ELEMENT_CLASS (parent_class)->change_state)
- return GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
- return GST_STATE_CHANGE_SUCCESS;
+
+ return ret;
}
static void