summaryrefslogtreecommitdiffstats
path: root/gst/modplug/gstmodplug.cc
diff options
context:
space:
mode:
Diffstat (limited to 'gst/modplug/gstmodplug.cc')
-rw-r--r--gst/modplug/gstmodplug.cc866
1 files changed, 439 insertions, 427 deletions
diff --git a/gst/modplug/gstmodplug.cc b/gst/modplug/gstmodplug.cc
index ec401793..6c677252 100644
--- a/gst/modplug/gstmodplug.cc
+++ b/gst/modplug/gstmodplug.cc
@@ -25,6 +25,21 @@
Olivier Lapicque <olivierl@jps.net>
*/
+/*
+ * SECTION:element-modplug
+ *
+ * <refsect2>
+ * <para>
+ * Modplug uses the <ulink url="http://modplug-xmms.sourceforge.net/">modplug</ulink>
+ * library to decode tracked music in the MOD/S3M/XM/IT and related formats.
+ * </para>
+ * <title>Example pipeline</title>
+ * <programlisting>
+ * gst-launch -v filesrc location=1990s-nostalgia.xm ! modplug ! audioconvert ! alsasink
+ * </programlisting>
+ * </refsect2>
+ */
+
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
@@ -38,6 +53,9 @@
#include <stdlib.h>
#include <gst/audio/audio.h>
+GST_DEBUG_CATEGORY_STATIC (modplug_debug);
+#define GST_CAT_DEFAULT modplug_debug
+
/* elementfactory information */
GstElementDetails modplug_details = {
"ModPlug",
@@ -46,14 +64,6 @@ GstElementDetails modplug_details = {
"Jeremy SIMON <jsimon13@yahoo.fr>"
};
-
-/* Filter signals and args */
-enum
-{
- /* FILL ME */
- LAST_SIGNAL
-};
-
enum
{
ARG_0,
@@ -71,87 +81,78 @@ enum
ARG_OVERSAMP
};
+#define DEFAULT_REVERB FALSE
+#define DEFAULT_REVERB_DEPTH 30
+#define DEFAULT_REVERB_DELAY 100
+#define DEFAULT_MEGABASS FALSE
+#define DEFAULT_MEGABASS_AMOUNT 40
+#define DEFAULT_MEGABASS_RANGE 30
+#define DEFAULT_SURROUND TRUE
+#define DEFAULT_SURROUND_DEPTH 20
+#define DEFAULT_SURROUND_DELAY 20
+#define DEFAULT_OVERSAMP TRUE
+#define DEFAULT_NOISE_REDUCTION TRUE
+
+#define SRC_CAPS \
+ "audio/x-raw-int," \
+ " endianness = (int) BYTE_ORDER," \
+ " signed = (boolean) true," \
+ " width = (int) 16," \
+ " depth = (int) 16," \
+ " rate = (int) { 8000, 11025, 22050, 44100 }," \
+ " channels = (int) 2; " \
+ "audio/x-raw-int," \
+ " endianness = (int) BYTE_ORDER," \
+ " signed = (boolean) false," \
+ " width = (int) 8," \
+ " depth = (int) 8," \
+ " rate = (int) { 8000, 11025, 22050, 44100 }, " \
+ " channels = (int) [ 1, 2 ]"
+
static GstStaticPadTemplate modplug_src_template_factory =
- GST_STATIC_PAD_TEMPLATE ("src",
+GST_STATIC_PAD_TEMPLATE ("src",
GST_PAD_SRC,
GST_PAD_ALWAYS,
- GST_STATIC_CAPS ("audio/x-raw-int, " "endianness = (int) BYTE_ORDER, " "signed = (boolean) TRUE, " "width = (int) 16, " "depth = (int) 16, " "rate = (int) { 8000, 11025, 22050, 44100 }, " /* FIXME? */
- "channels = (int) [ 1, 2 ]; " "audio/x-raw-int, " "endianness = (int) BYTE_ORDER, " "signed = (boolean) FALSE, " "width = (int) 8, " "depth = (int) 8, " "rate = (int) { 8000, 11025, 22050, 44100 }, " /* FIXME? */
- "channels = (int) [ 1, 2 ]")
- );
+ GST_STATIC_CAPS (SRC_CAPS));
static GstStaticPadTemplate modplug_sink_template_factory =
GST_STATIC_PAD_TEMPLATE ("sink",
GST_PAD_SINK,
GST_PAD_ALWAYS,
- GST_STATIC_CAPS ("audio/x-mod")
- );
-
-enum
-{
- MODPLUG_STATE_NEED_TUNE = 1,
- MODPLUG_STATE_LOAD_TUNE = 2,
- MODPLUG_STATE_PLAY_TUNE = 3
-};
+ GST_STATIC_CAPS ("audio/x-mod"));
-static void gst_modplug_base_init (GstModPlugClass * klass);
-static void gst_modplug_class_init (GstModPlugClass * klass);
-static void gst_modplug_init (GstModPlug * filter);
+static void gst_modplug_dispose (GObject * object);
static void gst_modplug_set_property (GObject * object,
guint id, const GValue * value, GParamSpec * pspec);
static void gst_modplug_get_property (GObject * object,
guint id, GValue * value, GParamSpec * pspec);
-static GstPadLinkReturn
-gst_modplug_srclink (GstPad * pad, const GstCaps * caps);
-static GstCaps *gst_modplug_fixate (GstPad * pad, const GstCaps * caps);
-static void gst_modplug_loop (GstElement * element);
-static void gst_modplug_setup (GstModPlug * modplug);
-static const GstFormat *gst_modplug_get_formats (GstPad * pad);
+
+static void gst_modplug_fixate (GstPad * pad, GstCaps * caps);
static const GstQueryType *gst_modplug_get_query_types (GstPad * pad);
static gboolean gst_modplug_src_event (GstPad * pad, GstEvent * event);
-static gboolean gst_modplug_src_query (GstPad * pad,
- GstQueryType type, GstFormat * format, gint64 * value);
+static gboolean gst_modplug_src_query (GstPad * pad, GstQuery * query);
static GstStateChangeReturn gst_modplug_change_state (GstElement * element,
GstStateChange transition);
-static GstElementClass *parent_class = NULL;
+static gboolean gst_modplug_sinkpad_activate (GstPad * pad);
+static gboolean gst_modplug_sinkpad_activate_pull (GstPad * pad,
+ gboolean active);
+static void gst_modplug_loop (GstModPlug * element);
-GType
-gst_modplug_get_type (void)
-{
- static GType modplug_type = 0;
-
- if (!modplug_type) {
- static const GTypeInfo modplug_info = {
- sizeof (GstModPlugClass),
- (GBaseInitFunc) gst_modplug_base_init,
- NULL,
- (GClassInitFunc) gst_modplug_class_init,
- NULL,
- NULL,
- sizeof (GstModPlug),
- 0,
- (GInstanceInitFunc) gst_modplug_init,
- NULL
- };
-
- modplug_type =
- g_type_register_static (GST_TYPE_ELEMENT, "GstModPlug", &modplug_info,
- (GTypeFlags) 0);
- }
- return modplug_type;
-}
+GST_BOILERPLATE (GstModPlug, gst_modplug, GstElement, GST_TYPE_ELEMENT);
static void
-gst_modplug_base_init (GstModPlugClass * klass)
+gst_modplug_base_init (gpointer g_class)
{
- GstElementClass *element_class = GST_ELEMENT_CLASS (klass);
+ GstElementClass *element_class = GST_ELEMENT_CLASS (g_class);
gst_element_class_add_pad_template (element_class,
gst_static_pad_template_get (&modplug_sink_template_factory));
gst_element_class_add_pad_template (element_class,
gst_static_pad_template_get (&modplug_src_template_factory));
gst_element_class_set_details (element_class, &modplug_details);
+
+ GST_DEBUG_CATEGORY_INIT (modplug_debug, "modplug", 0, "ModPlug element");
}
static void
@@ -163,159 +164,122 @@ gst_modplug_class_init (GstModPlugClass * klass)
gobject_class = (GObjectClass *) klass;
gstelement_class = (GstElementClass *) klass;
- parent_class = GST_ELEMENT_CLASS (g_type_class_ref (GST_TYPE_ELEMENT));
+ gobject_class->set_property = gst_modplug_set_property;
+ gobject_class->get_property = gst_modplug_get_property;
+ gobject_class->dispose = gst_modplug_dispose;
g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_SONGNAME,
g_param_spec_string ("songname", "Songname", "The song name",
- "", G_PARAM_READABLE));
+ NULL, G_PARAM_READABLE));
g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_REVERB,
- g_param_spec_boolean ("reverb", "reverb", "reverb",
- FALSE, (GParamFlags) G_PARAM_READWRITE));
+ g_param_spec_boolean ("reverb", "reverb", "Reverb",
+ DEFAULT_REVERB, (GParamFlags) G_PARAM_READWRITE));
g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_REVERB_DEPTH,
- g_param_spec_int ("reverb_depth", "reverb_depth", "reverb_depth",
- 0, 100, 30, (GParamFlags) G_PARAM_READWRITE));
+ g_param_spec_int ("reverb-depth", "reverb depth", "Reverb depth",
+ 0, 100, DEFAULT_REVERB_DEPTH, (GParamFlags) G_PARAM_READWRITE));
g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_REVERB_DELAY,
- g_param_spec_int ("reverb_delay", "reverb_delay", "reverb_delay",
- 0, 200, 100, (GParamFlags) G_PARAM_READWRITE));
+ g_param_spec_int ("reverb-delay", "reverb delay", "Reverb delay",
+ 0, 200, DEFAULT_REVERB_DELAY, (GParamFlags) G_PARAM_READWRITE));
g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_MEGABASS,
- g_param_spec_boolean ("megabass", "megabass", "megabass",
- FALSE, (GParamFlags) G_PARAM_READWRITE));
+ g_param_spec_boolean ("megabass", "megabass", "Megabass",
+ DEFAULT_MEGABASS, (GParamFlags) G_PARAM_READWRITE));
g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_MEGABASS_AMOUNT,
- g_param_spec_int ("megabass_amount", "megabass_amount", "megabass_amount",
- 0, 100, 40, (GParamFlags) G_PARAM_READWRITE));
+ g_param_spec_int ("megabass-amount", "megabass amount", "Megabass amount",
+ 0, 100, DEFAULT_MEGABASS_AMOUNT, (GParamFlags) G_PARAM_READWRITE));
g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_MEGABASS_RANGE,
- g_param_spec_int ("megabass_range", "megabass_range", "megabass_range",
- 0, 100, 30, (GParamFlags) G_PARAM_READWRITE));
+ g_param_spec_int ("megabass-range", "megabass range", "Megabass range",
+ 0, 100, DEFAULT_MEGABASS_RANGE, (GParamFlags) G_PARAM_READWRITE));
g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_SURROUND,
- g_param_spec_boolean ("surround", "surround", "surround",
- TRUE, (GParamFlags) G_PARAM_READWRITE));
+ g_param_spec_boolean ("surround", "surround", "Surround",
+ DEFAULT_SURROUND, (GParamFlags) G_PARAM_READWRITE));
g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_SURROUND_DEPTH,
- g_param_spec_int ("surround_depth", "surround_depth", "surround_depth",
- 0, 100, 20, (GParamFlags) G_PARAM_READWRITE));
+ g_param_spec_int ("surround-depth", "surround depth", "Surround depth",
+ 0, 100, DEFAULT_SURROUND_DEPTH, (GParamFlags) G_PARAM_READWRITE));
g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_SURROUND_DELAY,
- g_param_spec_int ("surround_delay", "surround_delay", "surround_delay",
- 0, 40, 20, (GParamFlags) G_PARAM_READWRITE));
+ g_param_spec_int ("surround-delay", "surround delay", "Surround delay",
+ 0, 40, DEFAULT_SURROUND_DELAY, (GParamFlags) G_PARAM_READWRITE));
g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_OVERSAMP,
g_param_spec_boolean ("oversamp", "oversamp", "oversamp",
- TRUE, (GParamFlags) G_PARAM_READWRITE));
+ DEFAULT_OVERSAMP, (GParamFlags) G_PARAM_READWRITE));
g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_NOISE_REDUCTION,
- g_param_spec_boolean ("noise_reduction", "noise_reduction",
- "noise_reduction", TRUE, (GParamFlags) G_PARAM_READWRITE));
-
- gobject_class->set_property = gst_modplug_set_property;
- gobject_class->get_property = gst_modplug_get_property;
+ g_param_spec_boolean ("noise-reduction", "noise reduction",
+ "noise reduction", DEFAULT_NOISE_REDUCTION,
+ (GParamFlags) G_PARAM_READWRITE));
- gstelement_class->change_state = gst_modplug_change_state;
+ gstelement_class->change_state = GST_DEBUG_FUNCPTR (gst_modplug_change_state);
}
static void
-gst_modplug_init (GstModPlug * modplug)
+gst_modplug_init (GstModPlug * modplug, GstModPlugClass * klass)
{
+ /* create the sink and src pads */
modplug->sinkpad =
- gst_pad_new_from_template (gst_static_pad_template_get
- (&modplug_sink_template_factory), "sink");
+ gst_pad_new_from_static_template (&modplug_sink_template_factory, "sink");
+ gst_pad_set_activate_function (modplug->sinkpad,
+ GST_DEBUG_FUNCPTR (gst_modplug_sinkpad_activate));
+ gst_pad_set_activatepull_function (modplug->sinkpad,
+ GST_DEBUG_FUNCPTR (gst_modplug_sinkpad_activate_pull));
gst_element_add_pad (GST_ELEMENT (modplug), modplug->sinkpad);
modplug->srcpad =
- gst_pad_new_from_template (gst_static_pad_template_get
- (&modplug_src_template_factory), "src");
- gst_pad_set_link_function (modplug->srcpad, gst_modplug_srclink);
- gst_pad_set_fixate_function (modplug->srcpad, gst_modplug_fixate);
+ gst_pad_new_from_static_template (&modplug_src_template_factory, "src");
+ gst_pad_set_fixatecaps_function (modplug->srcpad,
+ GST_DEBUG_FUNCPTR (gst_modplug_fixate));
gst_pad_set_event_function (modplug->srcpad,
- (GstPadEventFunction) GST_DEBUG_FUNCPTR (gst_modplug_src_event));
- gst_pad_set_query_function (modplug->srcpad, gst_modplug_src_query);
- gst_pad_set_query_type_function (modplug->srcpad, (GstPadQueryTypeFunction)
+ GST_DEBUG_FUNCPTR (gst_modplug_src_event));
+ gst_pad_set_query_function (modplug->srcpad,
+ GST_DEBUG_FUNCPTR (gst_modplug_src_query));
+ gst_pad_set_query_type_function (modplug->srcpad,
GST_DEBUG_FUNCPTR (gst_modplug_get_query_types));
- gst_pad_set_formats_function (modplug->srcpad,
- (GstPadFormatsFunction) GST_DEBUG_FUNCPTR (gst_modplug_get_formats));
gst_element_add_pad (GST_ELEMENT (modplug), modplug->srcpad);
- gst_element_set_loop_function (GST_ELEMENT (modplug), gst_modplug_loop);
-
- modplug->reverb = FALSE;
- modplug->reverb_depth = 30;
- modplug->reverb_delay = 100;
- modplug->megabass = FALSE;
- modplug->megabass_amount = 40;
- modplug->megabass_range = 30;
- modplug->surround = TRUE;
- modplug->surround_depth = 20;
- modplug->surround_delay = 20;
- modplug->oversamp = TRUE;
- modplug->noise_reduction = TRUE;
+ modplug->reverb = DEFAULT_REVERB;
+ modplug->reverb_depth = DEFAULT_REVERB_DEPTH;
+ modplug->reverb_delay = DEFAULT_REVERB_DELAY;
+ modplug->megabass = DEFAULT_MEGABASS;
+ modplug->megabass_amount = DEFAULT_MEGABASS_AMOUNT;
+ modplug->megabass_range = DEFAULT_MEGABASS_RANGE;
+ modplug->surround = DEFAULT_SURROUND;
+ modplug->surround_depth = DEFAULT_SURROUND_DEPTH;
+ modplug->surround_delay = DEFAULT_SURROUND_DELAY;
+ modplug->oversamp = DEFAULT_OVERSAMP;
+ modplug->noise_reduction = DEFAULT_NOISE_REDUCTION;
modplug->_16bit = TRUE;
modplug->channel = 2;
modplug->frequency = 44100;
- modplug->audiobuffer = NULL;
- modplug->buffer_in = NULL;
-
- modplug->state = MODPLUG_STATE_NEED_TUNE;
- GST_FLAG_SET (modplug, GST_ELEMENT_EVENT_AWARE);
}
+
static void
-gst_modplug_setup (GstModPlug * modplug)
+gst_modplug_dispose (GObject * object)
{
- if (modplug->_16bit)
- modplug->mSoundFile->SetWaveConfig (modplug->frequency, 16,
- modplug->channel);
- else
- modplug->mSoundFile->SetWaveConfig (modplug->frequency, 8,
- modplug->channel);
-
- modplug->mSoundFile->SetWaveConfigEx (modplug->surround, !modplug->oversamp,
- modplug->reverb, true, modplug->megabass, modplug->noise_reduction, true);
- modplug->mSoundFile->SetResamplingMode (SRCMODE_POLYPHASE);
-
- if (modplug->surround)
- modplug->mSoundFile->SetSurroundParameters (modplug->surround_depth,
- modplug->surround_delay);
-
- if (modplug->megabass)
- modplug->mSoundFile->SetXBassParameters (modplug->megabass_amount,
- modplug->megabass_range);
-
- if (modplug->reverb)
- modplug->mSoundFile->SetReverbParameters (modplug->reverb_depth,
- modplug->reverb_delay);
+ GstModPlug *modplug = GST_MODPLUG (object);
-}
+ G_OBJECT_CLASS (parent_class)->dispose (object);
-static const GstFormat *
-gst_modplug_get_formats (GstPad * pad)
-{
- static const GstFormat src_formats[] = {
-/* GST_FORMAT_BYTES,
- GST_FORMAT_DEFAULT,*/
- GST_FORMAT_TIME,
- (GstFormat) 0
- };
- static const GstFormat sink_formats[] = {
- /*GST_FORMAT_BYTES, */
- GST_FORMAT_TIME,
- (GstFormat) 0
- };
-
- return (GST_PAD_IS_SRC (pad) ? src_formats : sink_formats);
+ if (modplug->buffer) {
+ gst_buffer_unref (modplug->buffer);
+ modplug->buffer = NULL;
+ }
}
static const GstQueryType *
gst_modplug_get_query_types (GstPad * pad)
{
static const GstQueryType gst_modplug_src_query_types[] = {
- GST_QUERY_TOTAL,
+ GST_QUERY_DURATION,
GST_QUERY_POSITION,
(GstQueryType) 0
};
@@ -325,351 +289,409 @@ gst_modplug_get_query_types (GstPad * pad)
static gboolean
-gst_modplug_src_query (GstPad * pad, GstQueryType type,
- GstFormat * format, gint64 * value)
+gst_modplug_src_query (GstPad * pad, GstQuery * query)
{
- gboolean res = TRUE;
GstModPlug *modplug;
- gfloat tmp;
+ gboolean res = FALSE;
modplug = GST_MODPLUG (gst_pad_get_parent (pad));
- switch (type) {
- case GST_QUERY_TOTAL:
- switch (*format) {
- case GST_FORMAT_TIME:
- *value = (gint64) modplug->mSoundFile->GetSongTime () * GST_SECOND;
- break;
- default:
- res = FALSE;
- break;
+ if (!modplug->mSoundFile)
+ goto done;
+
+ switch (GST_QUERY_TYPE (query)) {
+ case GST_QUERY_DURATION:
+ {
+ GstFormat format;
+
+ gst_query_parse_duration (query, &format, NULL);
+ if (format == GST_FORMAT_TIME) {
+ gst_query_set_duration (query, format, modplug->song_length);
+ res = TRUE;
}
+ }
break;
case GST_QUERY_POSITION:
- switch (*format) {
- case GST_FORMAT_TIME:
- tmp =
- ((float) (modplug->mSoundFile->GetSongTime () *
- modplug->mSoundFile->GetCurrentPos ()) /
- (float) modplug->mSoundFile->GetMaxPosition ());
- *value = (gint64) (tmp * GST_SECOND);
- break;
- default:
- res = FALSE;
- break;
+ {
+ GstFormat format;
+
+ gst_query_parse_position (query, &format, NULL);
+ if (format == GST_FORMAT_TIME) {
+ gfloat fpos;
+ gint64 pos;
+
+ fpos = ((float) (modplug->song_length *
+ modplug->mSoundFile->GetCurrentPos ()) /
+ (float) modplug->mSoundFile->GetMaxPosition ());
+ pos = (gint64) (fpos * GST_SECOND);
+ gst_query_set_position (query, format, pos);
+ res = TRUE;
}
+ }
break;
default:
- res = FALSE;
+ res = gst_pad_query_default (pad, query);
break;
}
+done:
+ gst_object_unref (modplug);
+
return res;
}
-
static gboolean
gst_modplug_src_event (GstPad * pad, GstEvent * event)
{
- gboolean res = TRUE;
GstModPlug *modplug;
+ gboolean res = FALSE;
modplug = GST_MODPLUG (gst_pad_get_parent (pad));
switch (GST_EVENT_TYPE (event)) {
- /* the all-formats seek logic */
case GST_EVENT_SEEK:
{
- gboolean flush;
+ gdouble rate;
GstFormat format;
+ GstSeekFlags flags;
+ GstSeekType cur_type, stop_type;
+ gboolean flush;
+ gint64 cur, stop;
+ gint64 timestamp;
- format = GST_FORMAT_TIME;
+ if (modplug->frequency == 0) {
+ GST_DEBUG_OBJECT (modplug, "no song loaded yet");
+ break;
+ }
- /* shave off the flush flag, we'll need it later */
- flush = GST_EVENT_SEEK_FLAGS (event) & GST_SEEK_FLAG_FLUSH;
+ timestamp = gst_util_uint64_scale_int (modplug->offset, GST_SECOND,
+ modplug->frequency);
- modplug->seek_at = GST_EVENT_SEEK_OFFSET (event);
- break;
- }
- default:
- res = FALSE;
- break;
- }
+ gst_event_parse_seek (event, &rate, &format, &flags,
+ &cur_type, &cur, &stop_type, &stop);
- gst_event_unref (event);
+ if (format != GST_FORMAT_TIME) {
+ GST_DEBUG_OBJECT (modplug, "seeking is only supported in TIME format");
+ gst_event_unref (event);
+ break;
+ }
- return res;
-}
+ /* FIXME: we should be using GstSegment for all this */
+ if (cur_type != GST_SEEK_TYPE_SET || stop_type != GST_SEEK_TYPE_NONE) {
+ GST_DEBUG_OBJECT (modplug, "unsupported seek type");
+ gst_event_unref (event);
+ break;
+ }
-#if 0
-static GstCaps *
-gst_modplug_get_streaminfo (GstModPlug * modplug)
-{
- GstCaps *caps;
+ if (stop_type == GST_SEEK_TYPE_NONE)
+ stop = GST_CLOCK_TIME_NONE;
- props = gst_props_empty_new ();
+ cur = CLAMP (cur, 0, modplug->song_length);
- entry =
- gst_props_entry_new ("Patterns",
- G_TYPE_INT ((gint) modplug->mSoundFile->GetNumPatterns ()));
- gst_props_add_entry (props, (GstPropsEntry *) entry);
+ GST_DEBUG_OBJECT (modplug, "seek to %" GST_TIME_FORMAT,
+ GST_TIME_ARGS (cur));
- caps = gst_caps_new_simple ("application/x-gst-streaminfo", NULL);
- return caps;
-}
+ modplug->seek_at = cur;
+ flush = ((flags & GST_SEEK_FLAG_FLUSH) == GST_SEEK_FLAG_FLUSH);
-static void
-gst_modplug_update_info (GstModPlug * modplug)
-{
- if (modplug->streaminfo) {
- gst_caps_unref (modplug->streaminfo);
+ if (flush) {
+ gst_pad_push_event (modplug->srcpad, gst_event_new_flush_start ());
+ } else {
+ gst_pad_stop_task (modplug->sinkpad);
+ }
+
+ GST_PAD_STREAM_LOCK (modplug->sinkpad);
+
+ if (flags & GST_SEEK_FLAG_SEGMENT) {
+ gst_element_post_message (GST_ELEMENT (modplug),
+ gst_message_new_segment_start (GST_OBJECT (modplug), format, cur));
+ }
+ if (stop == -1 && modplug->song_length > 0)
+ stop = modplug->song_length;
+
+ if (flush) {
+ gst_pad_push_event (modplug->srcpad, gst_event_new_flush_stop ());
+ }
+
+ GST_LOG_OBJECT (modplug, "sending newsegment from %" GST_TIME_FORMAT "-%"
+ GST_TIME_FORMAT ", pos=%" GST_TIME_FORMAT, GST_TIME_ARGS (cur),
+ GST_TIME_ARGS (stop), GST_TIME_ARGS (cur));
+
+ gst_pad_push_event (modplug->srcpad,
+ gst_event_new_new_segment (FALSE, rate,
+ GST_FORMAT_TIME, cur, stop, cur));
+
+ modplug->offset =
+ gst_util_uint64_scale_int (cur, modplug->frequency, GST_SECOND);
+
+ gst_pad_start_task (modplug->sinkpad,
+ (GstTaskFunction) gst_modplug_loop, modplug);
+
+ GST_PAD_STREAM_UNLOCK (modplug->sinkpad);
+ res = TRUE;
+ break;
+ }
+ default:
+ res = gst_pad_event_default (pad, event);
+ break;
}
- modplug->streaminfo = gst_modplug_get_streaminfo (modplug);
- g_object_notify (G_OBJECT (modplug), "streaminfo");
+ gst_object_unref (modplug);
+ return res;
}
static void
-gst_modplug_update_metadata (GstModPlug * modplug)
+gst_modplug_fixate (GstPad * pad, GstCaps * caps)
{
- GstProps *props;
- GstPropsEntry *entry;
- const gchar *title;
-
- props = gst_props_empty_new ();
-
- title = modplug->mSoundFile->GetTitle ();
- entry = gst_props_entry_new ("Title", G_TYPE_STRING (title));
- gst_props_add_entry (props, entry);
-
- modplug->metadata = gst_caps_new_simple ("application/x-gst-metadata", NULL);
+ GstStructure *structure;
- g_object_notify (G_OBJECT (modplug), "metadata");
+ structure = gst_caps_get_structure (caps, 0);
+ if (gst_structure_fixate_field_nearest_int (structure, "rate", 44100))
+ return;
+ if (gst_structure_fixate_field_nearest_int (structure, "channels", 2))
+ return;
}
-#endif
-static GstPadLinkReturn
-gst_modplug_srclink (GstPad * pad, const GstCaps * caps)
+static gboolean
+gst_modplug_load_song (GstModPlug * modplug)
{
- GstModPlug *modplug;
+ GstCaps *newcaps, *othercaps;
GstStructure *structure;
gint depth;
- modplug = GST_MODPLUG (gst_pad_get_parent (pad));
+ GST_DEBUG_OBJECT (modplug, "Loading song");
- structure = gst_caps_get_structure (caps, 0);
+ modplug->mSoundFile = new CSoundFile;
+ /* negotiate srcpad caps */
+ othercaps = gst_pad_get_allowed_caps (modplug->srcpad);
+ structure = gst_caps_get_structure (othercaps, 0);
+ newcaps = gst_caps_copy_nth (othercaps, 0);
+ gst_caps_unref (othercaps);
+ gst_pad_fixate_caps (modplug->srcpad, newcaps);
+ gst_pad_set_caps (modplug->srcpad, newcaps);
+
+ /* set up modplug to output the negotiated format */
+ structure = gst_caps_get_structure (newcaps, 0);
gst_structure_get_int (structure, "depth", &depth);
modplug->_16bit = (depth == 16);
gst_structure_get_int (structure, "channels", &modplug->channel);
gst_structure_get_int (structure, "rate", &modplug->frequency);
- modplug->length = 1152 * modplug->channel * depth / 8;
- gst_modplug_setup (modplug);
+ modplug->read_samples = 1152;
+ modplug->read_bytes = modplug->read_samples * modplug->channel * depth / 8;
+
+ if (modplug->_16bit)
+ modplug->mSoundFile->SetWaveConfig (modplug->frequency, 16,
+ modplug->channel);
+ else
+ modplug->mSoundFile->SetWaveConfig (modplug->frequency, 8,
+ modplug->channel);
+
+ modplug->mSoundFile->SetWaveConfigEx (modplug->surround, !modplug->oversamp,
+ modplug->reverb, true, modplug->megabass, modplug->noise_reduction, true);
+ modplug->mSoundFile->SetResamplingMode (SRCMODE_POLYPHASE);
- return GST_PAD_LINK_OK;
+ if (modplug->surround)
+ modplug->mSoundFile->SetSurroundParameters (modplug->surround_depth,
+ modplug->surround_delay);
+
+ if (modplug->megabass)
+ modplug->mSoundFile->SetXBassParameters (modplug->megabass_amount,
+ modplug->megabass_range);
+
+ if (modplug->reverb)
+ modplug->mSoundFile->SetReverbParameters (modplug->reverb_depth,
+ modplug->reverb_delay);
+
+ if (!modplug->mSoundFile->Create (GST_BUFFER_DATA (modplug->buffer),
+ modplug->song_size)) {
+ GST_ELEMENT_ERROR (modplug, STREAM, DECODE, (NULL),
+ ("Unable to load song"));
+ return FALSE;
+ }
+
+ modplug->song_length = modplug->mSoundFile->GetSongTime () * GST_SECOND;
+ modplug->seek_at = -1;
+
+ GST_INFO_OBJECT (modplug, "Song length: %" GST_TIME_FORMAT,
+ GST_TIME_ARGS (modplug->song_length));
+
+ return TRUE;
+}
+
+static gboolean
+gst_modplug_sinkpad_activate (GstPad * pad)
+{
+ if (!gst_pad_check_pull_range (pad))
+ return FALSE;
+
+ return gst_pad_activate_pull (pad, TRUE);
}
-static GstCaps *
-gst_modplug_fixate (GstPad * pad, const GstCaps * caps)
+static gboolean
+gst_modplug_sinkpad_activate_pull (GstPad * pad, gboolean active)
{
- if (gst_caps_is_simple (caps)) {
- GstCaps *copy;
- GstStructure *structure;
-
- copy = gst_caps_copy (caps);
- structure = gst_caps_get_structure (copy, 0);
- if (gst_structure_fixate_field_nearest_int (structure, "rate", 44100))
- return copy;
- if (gst_structure_fixate_field_nearest_int (structure, "channels", 2))
- return copy;
- gst_caps_free (copy);
+ GstModPlug *modplug = GST_MODPLUG (GST_OBJECT_PARENT (pad));
+
+ if (active) {
+ return gst_pad_start_task (pad, (GstTaskFunction) gst_modplug_loop,
+ modplug);
+ } else {
+ return gst_pad_stop_task (pad);
}
- return NULL;
}
-static void
-gst_modplug_handle_event (GstModPlug * modplug)
+static gboolean
+gst_modplug_get_upstream_size (GstModPlug * modplug, gint64 * length)
{
- gint64 value;
- guint32 remaining;
- GstEvent *event;
+ GstFormat format = GST_FORMAT_BYTES;
+ gboolean res = FALSE;
+ GstPad *peer;
- gst_bytestream_get_status (modplug->bs, &remaining, &event);
+ peer = gst_pad_get_peer (modplug->sinkpad);
+ if (peer == NULL)
+ return FALSE;
- if (!event) {
- g_warning ("modplug: no bytestream event");
- return;
+ if (gst_pad_query_duration (peer, &format, length) && *length >= 0) {
+ res = TRUE;
}
- switch (GST_EVENT_TYPE (event)) {
- case GST_EVENT_EOS:
- gst_event_unref (event);
- break;
- case GST_EVENT_DISCONTINUOUS:
- if (gst_event_discont_get_value (event, GST_FORMAT_BYTES, &value)) {
- if (remaining == value) {
- gst_event_unref (event);
- break;
- }
- }
- gst_bytestream_flush_fast (modplug->bs, remaining);
- default:
- gst_pad_event_default (modplug->sinkpad, event);
- break;
- }
+ gst_object_unref (peer);
+ return res;
}
static void
-gst_modplug_loop (GstElement * element)
+gst_modplug_loop (GstModPlug * modplug)
{
- GstModPlug *modplug;
- GstEvent *event;
-
- g_return_if_fail (element != NULL);
- g_return_if_fail (GST_IS_MODPLUG (element));
+ GstFlowReturn flow;
+ GstBuffer *out = NULL;
- modplug = GST_MODPLUG (element);
+ g_assert (GST_IS_MODPLUG (modplug));
- if (modplug->state == MODPLUG_STATE_NEED_TUNE) {
-/* GstBuffer *buf;*/
+ /* first, get the size of the song */
+ if (!modplug->song_size) {
+ if (!gst_modplug_get_upstream_size (modplug, &modplug->song_size)) {
+ GST_ELEMENT_ERROR (modplug, STREAM, DECODE, (NULL),
+ ("Unable to load song"));
+ goto pause;
+ }
- modplug->seek_at = -1;
- modplug->need_discont = TRUE;
- modplug->eos = FALSE;
-/*
- buf = gst_pad_pull (modplug->sinkpad);
- g_assert (buf != NULL);
-
- if (GST_IS_EVENT (buf)) {
- GstEvent *event = GST_EVENT (buf);
-
- switch (GST_EVENT_TYPE (buf)) {
- case GST_EVENT_EOS:
- modplug->state = MODPLUG_STATE_LOAD_TUNE;
- break;
- case GST_EVENT_DISCONTINUOUS:
- break;
- default:
- bail out, we're not going to do anything
- gst_event_unref (event);
- gst_pad_send_event (modplug->srcpad, gst_event_new (GST_EVENT_EOS));
- gst_element_set_eos (element);
- return;
- }
- gst_event_unref (event);
+ if (modplug->buffer) {
+ gst_buffer_unref (modplug->buffer);
}
- else {
- memcpy (modplug->buffer_in + modplug->song_size, GST_BUFFER_DATA (buf), GST_BUFFER_SIZE (buf));
- modplug->song_size += GST_BUFFER_SIZE (buf);
+ modplug->buffer = gst_buffer_new_and_alloc (modplug->song_size);
+ modplug->offset = 0;
+ }
- gst_buffer_unref (buf);
+ /* read in the song data */
+ if (!modplug->mSoundFile) {
+ GstBuffer *buffer = NULL;
+ guint64 read_size = modplug->song_size - modplug->offset;
+
+ if (read_size > 4096)
+ read_size = 4096;
+
+ flow =
+ gst_pad_pull_range (modplug->sinkpad, modplug->offset, read_size,
+ &buffer);
+ if (flow != GST_FLOW_OK) {
+ GST_ELEMENT_ERROR (modplug, STREAM, DECODE, (NULL),
+ ("Unable to load song"));
+ goto pause;
}
-*/
- if (modplug->bs) {
- guint64 got;
+ /* GST_LOG_OBJECT (modplug, "Read %u bytes", GST_BUFFER_SIZE (buffer)); */
+ g_memmove (GST_BUFFER_DATA (modplug->buffer) + modplug->offset,
+ GST_BUFFER_DATA (buffer), GST_BUFFER_SIZE (buffer));
+ gst_buffer_unref (buffer);
- modplug->song_size = gst_bytestream_length (modplug->bs);
+ modplug->offset += read_size;
- got =
- gst_bytestream_peek_bytes (modplug->bs, &modplug->buffer_in,
- modplug->song_size);
+ /* actually load it */
+ if (modplug->offset == modplug->song_size) {
+ GstEvent *newsegment;
+ gboolean ok;
- if (got < modplug->song_size) {
- gst_modplug_handle_event (modplug);
- return;
- }
- modplug->state = MODPLUG_STATE_LOAD_TUNE;
- }
- }
+ ok = gst_modplug_load_song (modplug);
+ gst_buffer_unref (modplug->buffer);
+ modplug->buffer = NULL;
+ modplug->offset = 0;
- if (modplug->state == MODPLUG_STATE_LOAD_TUNE) {
- modplug->mSoundFile = new CSoundFile;
+ if (!ok) {
+ goto pause;
+ }
- if (!GST_PAD_CAPS (modplug->srcpad) &&
- GST_PAD_LINK_FAILED (gst_pad_renegotiate (modplug->srcpad))) {
- GST_ELEMENT_ERROR (modplug, CORE, NEGOTIATION, (NULL), (NULL));
+ newsegment = gst_event_new_new_segment (FALSE, 1.0, GST_FORMAT_TIME,
+ 0, modplug->song_length, 0);
+ gst_pad_push_event (modplug->srcpad, newsegment);
+ } else {
+ /* not fully loaded yet */
return;
}
+ }
- modplug->mSoundFile->Create (modplug->buffer_in, modplug->song_size);
- modplug->opened = TRUE;
-
- gst_bytestream_flush (modplug->bs, modplug->song_size);
- modplug->buffer_in = NULL;
+ /* could move this to gst_modplug_src_event
+ * if libmodplug was definitely thread safe.. */
+ if (modplug->seek_at != -1) {
+ gint seek_to_pos;
+ gfloat temp;
- modplug->audiobuffer = (guchar *) g_malloc (modplug->length);
+ temp = (gfloat) modplug->song_length / modplug->seek_at;
+ seek_to_pos = (int) (modplug->mSoundFile->GetMaxPosition () / temp);
- //gst_modplug_update_metadata (modplug);
- //gst_modplug_update_info (modplug);
+ GST_DEBUG_OBJECT (modplug, "Seeking to row %d", seek_to_pos);
- modplug->state = MODPLUG_STATE_PLAY_TUNE;
+ modplug->mSoundFile->SetCurrentPos (seek_to_pos);
+ modplug->seek_at = -1;
}
- if (modplug->state == MODPLUG_STATE_PLAY_TUNE && !modplug->eos) {
- if (modplug->seek_at != -1) {
- gint seek_to_pos;
- gint64 total;
- gfloat temp;
+ /* read and output a buffer */
+ flow = gst_pad_alloc_buffer_and_set_caps (modplug->srcpad,
+ GST_BUFFER_OFFSET_NONE, modplug->read_bytes,
+ GST_PAD_CAPS (modplug->srcpad), &out);
- total = modplug->mSoundFile->GetSongTime () * GST_SECOND;
+ if (flow != GST_FLOW_OK) {
+ GST_LOG_OBJECT (modplug, "pad alloc flow: %s", gst_flow_get_name (flow));
+ goto pause;
+ }
- temp = (gfloat) total / modplug->seek_at;
- seek_to_pos = (int) (modplug->mSoundFile->GetMaxPosition () / temp);
+ if (!modplug->mSoundFile->Read (GST_BUFFER_DATA (out), modplug->read_bytes))
+ goto eos;
- modplug->mSoundFile->SetCurrentPos (seek_to_pos);
- modplug->need_discont = TRUE;
- modplug->seek_at = -1;
- }
+ GST_BUFFER_SIZE (out) = modplug->read_bytes;
+ GST_BUFFER_DURATION (out) =
+ gst_util_uint64_scale_int (modplug->read_samples, GST_SECOND,
+ modplug->frequency);
+ GST_BUFFER_OFFSET (out) = modplug->offset;
+ GST_BUFFER_TIMESTAMP (out) =
+ gst_util_uint64_scale_int (modplug->offset, GST_SECOND,
+ modplug->frequency);
- if (modplug->need_discont && GST_PAD_IS_USABLE (modplug->srcpad)) {
- GstEvent *discont;
- gint64 value;
- GstFormat format = GST_FORMAT_TIME;
-
- if (gst_modplug_src_query (modplug->srcpad, GST_QUERY_POSITION, &format,
- &value)) {
- discont =
- gst_event_new_discontinuous (FALSE, GST_FORMAT_TIME, value,
- GST_FORMAT_UNDEFINED);
- modplug->timestamp = value;
- } else {
- modplug->timestamp = GST_CLOCK_TIME_NONE;
- discont = gst_event_new_discontinuous (FALSE, GST_FORMAT_UNDEFINED);
- }
- gst_pad_push (modplug->srcpad, GST_DATA (discont));
- modplug->need_discont = FALSE;
- }
+ modplug->offset += modplug->read_samples;
- if (modplug->mSoundFile->Read (modplug->audiobuffer, modplug->length) != 0) {
- GstBuffer *buffer_out;
+ flow = gst_pad_push (modplug->srcpad, out);
- buffer_out = gst_buffer_new ();
- GST_BUFFER_DATA (buffer_out) =
- (guchar *) g_memdup (modplug->audiobuffer, modplug->length);
- GST_BUFFER_SIZE (buffer_out) = modplug->length;
- GST_BUFFER_TIMESTAMP (buffer_out) = modplug->timestamp;
+ if (flow != GST_FLOW_OK) {
+ GST_LOG_OBJECT (modplug, "pad push flow: %s", gst_flow_get_name (flow));
+ goto pause;
+ }
- if (GST_CLOCK_TIME_IS_VALID (modplug->timestamp)) {
- GST_BUFFER_DURATION (buffer_out) =
- modplug->length * GST_SECOND / modplug->frequency /
- modplug->channel / (modplug->_16bit ? 2 : 1);
- modplug->timestamp += GST_BUFFER_DURATION (buffer_out);
- }
+ return;
- if (GST_PAD_IS_USABLE (modplug->srcpad))
- gst_pad_push (modplug->srcpad, GST_DATA (buffer_out));
- } else if (GST_PAD_IS_LINKED (modplug->srcpad)) {
- /* FIXME, hack, pull final EOS from peer */
- gst_bytestream_flush (modplug->bs, 1);
+eos:
+ {
+ gst_buffer_unref (out);
+ GST_INFO_OBJECT (modplug, "EOS");
+ gst_pad_push_event (modplug->srcpad, gst_event_new_eos ());
+ goto pause;
+ }
- event = gst_event_new (GST_EVENT_EOS);
- gst_pad_push (modplug->srcpad, GST_DATA (event));
- gst_element_set_eos (element);
- modplug->eos = TRUE;
- }
+pause:
+ {
+ GST_INFO_OBJECT (modplug, "Pausing");
+ gst_pad_pause_task (modplug->sinkpad);
}
}
@@ -678,43 +700,39 @@ static GstStateChangeReturn
gst_modplug_change_state (GstElement * element, GstStateChange transition)
{
GstModPlug *modplug;
+ GstStateChangeReturn ret;
modplug = GST_MODPLUG (element);
switch (transition) {
- case GST_STATE_CHANGE_NULL_TO_READY:
- break;
case GST_STATE_CHANGE_READY_TO_PAUSED:
- modplug->bs = gst_bytestream_new (modplug->sinkpad);
+ modplug->buffer = NULL;
+ modplug->offset = 0;
modplug->song_size = 0;
- modplug->state = MODPLUG_STATE_NEED_TUNE;
- break;
- case GST_STATE_CHANGE_PAUSED_TO_PLAYING:
break;
- case GST_STATE_CHANGE_PLAYING_TO_PAUSED:
+ default:
break;
+ }
+
+ ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
+ if (ret == GST_STATE_CHANGE_FAILURE)
+ return ret;
+
+ switch (transition) {
case GST_STATE_CHANGE_PAUSED_TO_READY:
- gst_bytestream_destroy (modplug->bs);
- modplug->bs = NULL;
- if (modplug->opened) {
+ if (modplug->buffer) {
+ gst_buffer_unref (modplug->buffer);
+ modplug->buffer = NULL;
+ }
+ if (modplug->mSoundFile) {
modplug->mSoundFile->Destroy ();
- modplug->opened = FALSE;
+ modplug->mSoundFile = NULL;
}
- if (modplug->audiobuffer)
- g_free (modplug->audiobuffer);
- modplug->buffer_in = NULL;
- modplug->audiobuffer = NULL;
- modplug->state = MODPLUG_STATE_NEED_TUNE;
- break;
- case GST_STATE_CHANGE_READY_TO_NULL:
break;
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;
}
@@ -725,7 +743,6 @@ gst_modplug_set_property (GObject * object, guint id, const GValue * value,
{
GstModPlug *modplug;
- /* it's not null if we got it, but it might not be ours */
g_return_if_fail (GST_IS_MODPLUG (object));
modplug = GST_MODPLUG (object);
@@ -771,7 +788,6 @@ gst_modplug_get_property (GObject * object, guint id, GValue * value,
{
GstModPlug *modplug;
- /* it's not null if we got it, but it might not be ours */
g_return_if_fail (GST_IS_MODPLUG (object));
modplug = GST_MODPLUG (object);
@@ -814,10 +830,6 @@ gst_modplug_get_property (GObject * object, guint id, GValue * value,
static gboolean
plugin_init (GstPlugin * plugin)
{
- /* this filter needs the bytestream package */
- if (!gst_library_load ("gstbytestream"))
- return FALSE;
-
return gst_element_register (plugin, "modplug",
GST_RANK_PRIMARY, GST_TYPE_MODPLUG);
}