summaryrefslogtreecommitdiffstats
path: root/gst/spectrum/gstspectrum.c
diff options
context:
space:
mode:
Diffstat (limited to 'gst/spectrum/gstspectrum.c')
-rw-r--r--gst/spectrum/gstspectrum.c167
1 files changed, 102 insertions, 65 deletions
diff --git a/gst/spectrum/gstspectrum.c b/gst/spectrum/gstspectrum.c
index 4f01ceff..12762c58 100644
--- a/gst/spectrum/gstspectrum.c
+++ b/gst/spectrum/gstspectrum.c
@@ -1,5 +1,6 @@
/* GStreamer
* Copyright (C) <1999> Erik Walthinsen <omega@cse.ogi.edu>
+ * <2006> Stefan Kost <ensonic@users.sf.net>
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
@@ -24,12 +25,33 @@
#include "gstspectrum.h"
+GST_DEBUG_CATEGORY_STATIC (gst_spectrum_debug);
+#define GST_CAT_DEFAULT gst_spectrum_debug
+
/* elementfactory information */
static const GstElementDetails gst_spectrum_details =
GST_ELEMENT_DETAILS ("Spectrum analyzer",
"Filter/Analyzer/Audio",
"Run an FFT on the audio signal, output spectrum data",
- "Erik Walthinsen <omega@cse.ogi.edu>");
+ "Erik Walthinsen <omega@cse.ogi.edu>,"
+ "Stefan Kost <ensonic@users.sf.net>");
+
+static GstStaticPadTemplate sink_template_factory =
+GST_STATIC_PAD_TEMPLATE ("sink",
+ GST_PAD_SINK,
+ GST_PAD_ALWAYS,
+ GST_STATIC_CAPS ("audio/x-raw-int, "
+ "rate = (int) [ 1, MAX ], "
+ "channels = (int) 1, "
+ "endianness = (int) BYTE_ORDER, "
+ "width = (int) 16, " "depth = (int) 16, " "signed = (boolean) true")
+ );
+
+static GstStaticPadTemplate src_template_factory =
+GST_STATIC_PAD_TEMPLATE ("src",
+ GST_PAD_SRC,
+ GST_PAD_ALWAYS,
+ GST_STATIC_CAPS_ANY);
/* Spectrum signals and args */
enum
@@ -48,18 +70,17 @@ enum
static void gst_spectrum_base_init (gpointer g_class);
static void gst_spectrum_class_init (GstSpectrumClass * klass);
static void gst_spectrum_init (GstSpectrum * spectrum);
-
+static void gst_spectrum_dispose (GObject * object);
static void gst_spectrum_set_property (GObject * object, guint prop_id,
const GValue * value, GParamSpec * pspec);
-static void gst_spectrum_chain (GstPad * pad, GstData * _data);
+static GstFlowReturn gst_spectrum_chain (GstPad * pad, GstBuffer * buffer);
#define fixed short
-int gst_spectrum_fix_fft (fixed fr[], fixed fi[], int m, int inverse);
-void gst_spectrum_fix_loud (fixed loud[], fixed fr[], fixed fi[], int n,
+extern int gst_spectrum_fix_fft (fixed fr[], fixed fi[], int m, int inverse);
+extern void gst_spectrum_fix_loud (fixed loud[], fixed fr[], fixed fi[], int n,
int scale_shift);
-void gst_spectrum_window (fixed fr[], int n);
-
+extern void gst_spectrum_window (fixed fr[], int n);
static GstElementClass *parent_class = NULL;
@@ -83,9 +104,10 @@ gst_spectrum_get_type (void)
(GInstanceInitFunc) gst_spectrum_init,
};
- spectrum_type =
- g_type_register_static (GST_TYPE_ELEMENT, "GstSpectrum", &spectrum_info,
- 0);
+ spectrum_type = g_type_register_static (GST_TYPE_ELEMENT, "GstSpectrum",
+ &spectrum_info, 0);
+ GST_DEBUG_CATEGORY_INIT (gst_spectrum_debug, "spectrum", 0,
+ "audio spectrum analyser element");
}
return spectrum_type;
}
@@ -95,113 +117,128 @@ gst_spectrum_base_init (gpointer g_class)
{
GstElementClass *element_class = GST_ELEMENT_CLASS (g_class);
+ gst_element_class_add_pad_template (element_class,
+ gst_static_pad_template_get (&sink_template_factory));
+ gst_element_class_add_pad_template (element_class,
+ gst_static_pad_template_get (&src_template_factory));
gst_element_class_set_details (element_class, &gst_spectrum_details);
}
+
static void
gst_spectrum_class_init (GstSpectrumClass * klass)
{
- GObjectClass *gobject_class;
-
- gobject_class = (GObjectClass *) klass;
+ GObjectClass *gobject_class = (GObjectClass *) klass;
parent_class = g_type_class_peek_parent (klass);
- g_object_class_install_property (G_OBJECT_CLASS (klass), ARG_WIDTH, g_param_spec_int ("width", "width", "width", G_MININT, G_MAXINT, 0, G_PARAM_WRITABLE)); /* CHECKME */
-
gobject_class->set_property = gst_spectrum_set_property;
+ gobject_class->dispose = gst_spectrum_dispose;
+
+ g_object_class_install_property (gobject_class, ARG_WIDTH,
+ g_param_spec_int ("width", "width", "width",
+ G_MININT, G_MAXINT, 0, G_PARAM_WRITABLE));
+
}
static void
gst_spectrum_init (GstSpectrum * spectrum)
{
- spectrum->sinkpad = gst_pad_new ("sink", GST_PAD_SINK);
+ spectrum->sinkpad =
+ gst_pad_new_from_static_template (&sink_template_factory, "sink");
gst_element_add_pad (GST_ELEMENT (spectrum), spectrum->sinkpad);
gst_pad_set_chain_function (spectrum->sinkpad, gst_spectrum_chain);
- spectrum->srcpad = gst_pad_new ("src", GST_PAD_SRC);
+
+ spectrum->srcpad =
+ gst_pad_new_from_static_template (&src_template_factory, "src");
gst_element_add_pad (GST_ELEMENT (spectrum), spectrum->srcpad);
spectrum->width = 75;
+ spectrum->base = 8;
+ spectrum->len = 1024;
+
+ spectrum->loud = g_malloc (spectrum->len * sizeof (gint16));
+ spectrum->im = g_malloc (spectrum->len * sizeof (gint16));
+ memset (spectrum->im, 0, spectrum->len * sizeof (gint16));
+ spectrum->re = g_malloc (spectrum->len * sizeof (gint16));
+ memset (spectrum->re, 0, spectrum->len * sizeof (gint16));
+}
+
+static void
+gst_spectrum_dispose (GObject * object)
+{
+ GstSpectrum *spectrum = GST_SPECTRUM (object);
+
+ g_free (spectrum->re);
+ g_free (spectrum->im);
+ g_free (spectrum->loud);
+
+ G_OBJECT_CLASS (parent_class)->dispose (object);
}
static void
gst_spectrum_set_property (GObject * object, guint prop_id,
const GValue * value, GParamSpec * pspec)
{
- GstSpectrum *spectrum;
-
- g_return_if_fail (GST_IS_SPECTRUM (object));
- spectrum = GST_SPECTRUM (object);
+ GstSpectrum *spectrum = GST_SPECTRUM (object);
switch (prop_id) {
case ARG_WIDTH:
spectrum->width = g_value_get_int (value);
break;
default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break;
}
}
-static void
-gst_spectrum_chain (GstPad * pad, GstData * _data)
+static GstFlowReturn
+gst_spectrum_chain (GstPad * pad, GstBuffer * buffer)
{
- GstBuffer *buf = GST_BUFFER (_data);
GstSpectrum *spectrum;
- gint spec_base, spec_len;
- gint16 *re, *im, *loud;
gint16 *samples;
- gint step, pos, i;
+ gint step, pos, num, i;
guchar *spect;
GstBuffer *newbuf;
- g_return_if_fail (pad != NULL);
- g_return_if_fail (GST_IS_PAD (pad));
- g_return_if_fail (buf != NULL);
-
spectrum = GST_SPECTRUM (GST_OBJECT_PARENT (pad));
+ samples = (gint16 *) GST_BUFFER_DATA (buffer);
+
+ GST_LOG ("buffer-size = %ld", GST_BUFFER_SIZE (buffer));
+
+ /* FIXME:need a gst_adapter */
+ num = GST_BUFFER_SIZE (buffer) / 2;
+ num = MIN (num, spectrum->len);
+
+ for (i = 0; i < num; i++)
+ spectrum->re[i] = (samples[(i * 2)] + samples[(i * 2) + 1]) >> 1;
+
+ gst_spectrum_window (spectrum->re, spectrum->len);
+ gst_spectrum_fix_fft (spectrum->re, spectrum->im, spectrum->base, FALSE);
+ gst_spectrum_fix_loud (spectrum->loud, spectrum->re, spectrum->im,
+ spectrum->len, 0);
- samples = (gint16 *) GST_BUFFER_DATA (buf);
-
- spec_base = 8;
- spec_len = 1024;
-
- im = g_malloc (spec_len * sizeof (gint16));
- g_return_if_fail (im != NULL);
- loud = g_malloc (spec_len * sizeof (gint16));
- g_return_if_fail (loud != NULL);
-
- memset (im, 0, spec_len * sizeof (gint16));
- /*if (spectrum->meta->channels == 2) { */
- re = g_malloc (spec_len * sizeof (gint16));
- for (i = 0; i < spec_len; i++)
- re[i] = (samples[(i * 2)] + samples[(i * 2) + 1]) >> 1;
- /*} else */
- /* re = samples; */
- gst_spectrum_window (re, spec_len);
- gst_spectrum_fix_fft (re, im, spec_base, FALSE);
- gst_spectrum_fix_loud (loud, re, im, spec_len, 0);
- if (re != samples)
- g_free (re);
- g_free (im);
- step = spec_len / (spectrum->width * 2);
+ /* resample to requested width */
+ step = spectrum->len / (spectrum->width * 4); /* <-- shouldn't this be 2 instead of 4? */
spect = (guchar *) g_malloc (spectrum->width);
for (i = 0, pos = 0; i < spectrum->width; i++, pos += step) {
- if (loud[pos] > -60)
- spect[i] = (loud[pos] + 60) / 2;
- else
+ /* > -60 db? */
+ if (spectrum->loud[pos] > -60) {
+ spect[i] = spectrum->loud[pos] + 60;
+ /*
+ if (spect[i] > 15);
+ spect[i] = 15;
+ */
+ } else
+ /* treat as silence */
spect[i] = 0;
-/* if (spect[i] > 15); */
-/* spect[i] = 15; */
}
- g_free (loud);
- gst_buffer_unref (buf);
-/* g_free(samples); */
+ gst_buffer_unref (buffer);
newbuf = gst_buffer_new ();
- g_return_if_fail (newbuf != NULL);
GST_BUFFER_DATA (newbuf) = spect;
GST_BUFFER_SIZE (newbuf) = spectrum->width;
- gst_pad_push (spectrum->srcpad, GST_DATA (newbuf));
+ return gst_pad_push (spectrum->srcpad, newbuf);
}
static gboolean