/* -*- c-basic-offset: 2 -*- * GStreamer * Copyright (C) 1999-2001 Erik Walthinsen * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public * License along with this library; if not, write to the * Free Software Foundation, Inc., 59 Temple Place - Suite 330, * Boston, MA 02111-1307, USA. */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include #include #include #include #include #include "gstvolume.h" /* some defines for audio processing */ /* the volume factor is a range from 0.0 to (arbitrary) 4.0 * we map 1.0 to VOLUME_UNITY_INT */ #define VOLUME_UNITY_INT 8192 /* internal int for unity */ #define VOLUME_UNITY_BIT_SHIFT 13 /* number of bits to shift for unity */ #define VOLUME_MAX_DOUBLE 4.0 #define VOLUME_MAX_INT16 32767 #define VOLUME_MIN_INT16 -32768 /* number of steps we use for the mixer interface to go from 0.0 to 1.0 */ # define VOLUME_STEPS 100 static GstElementDetails volume_details = { "Volume", "Filter/Effect/Audio", "Set volume on audio/raw streams", "Andy Wingo ", }; /* Filter signals and args */ enum { /* FILL ME */ LAST_SIGNAL }; enum { ARG_0, ARG_SILENT, ARG_MUTE, ARG_VOLUME }; static GstStaticPadTemplate volume_sink_factory = GST_STATIC_PAD_TEMPLATE ( "sink", GST_PAD_SINK, GST_PAD_ALWAYS, GST_STATIC_CAPS ( "audio/x-raw-float, " "rate = (int) [ 1, MAX ], " "channels = (int) [ 1, MAX ], " "endianness = (int) BYTE_ORDER, " "width = (int) 32, " "buffer-frames = (int) [ 1, MAX]; " "audio/x-raw-int, " "channels = (int) [ 1, MAX ], " "rate = (int) [ 1, MAX ], " "endianness = (int) BYTE_ORDER, " "width = (int) 16, " "depth = (int) 16, " "signed = (bool) TRUE" ) ); static GstStaticPadTemplate volume_src_factory = GST_STATIC_PAD_TEMPLATE ( "src", GST_PAD_SRC, GST_PAD_ALWAYS, GST_STATIC_CAPS ( "audio/x-raw-float, " "rate = (int) [ 1, MAX ], " "channels = (int) [ 1, MAX ], " "endianness = (int) BYTE_ORDER, " "width = (int) 32, " "buffer-frames = (int) [ 1, MAX]; " "audio/x-raw-int, " "channels = (int) [ 1, MAX ], " "rate = (int) [ 1, MAX ], " "endianness = (int) BYTE_ORDER, " "width = (int) 16, " "depth = (int) 16, " "signed = (bool) TRUE" ) ); static void volume_base_init (gpointer g_class); static void volume_class_init (GstVolumeClass *klass); static void volume_init (GstVolume *filter); static void volume_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec); static void volume_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec); static void volume_update_volume (const GValue *value, gpointer data); static void volume_update_mute (const GValue *value, gpointer data); static gboolean volume_parse_caps (GstVolume *filter, GstStructure *structure); static void volume_chain_float (GstPad *pad, GstData *_data); static void volume_chain_int16 (GstPad *pad, GstData *_data); static void gst_volume_interface_init (GstImplementsInterfaceClass *klass); static void gst_volume_mixer_init (GstMixerClass *iface); static GstElementClass *parent_class = NULL; /*static guint gst_filter_signals[LAST_SIGNAL] = { 0 }; */ static gboolean gst_volume_interface_supported (GstImplementsInterface *iface, GType type) { g_assert (type == GST_TYPE_MIXER); return TRUE; } static void gst_volume_interface_init (GstImplementsInterfaceClass *klass) { klass->supported = gst_volume_interface_supported; } static const GList * gst_volume_list_tracks (GstMixer *mixer) { GstVolume *filter = GST_VOLUME (mixer); g_return_val_if_fail (filter != NULL, NULL); g_return_val_if_fail (GST_IS_VOLUME (filter), NULL); return filter->tracklist; } static void gst_volume_set_volume (GstMixer *mixer, GstMixerTrack *track, gint *volumes) { GstVolume *filter = GST_VOLUME (mixer); g_return_if_fail (filter != NULL); g_return_if_fail (GST_IS_VOLUME (filter)); gst_dpman_bypass_dparam (filter->dpman, "volume"); filter->volume_f = (gfloat) volumes[0] / VOLUME_STEPS; filter->volume_i = filter->volume_f * VOLUME_UNITY_INT; if (filter->mute) { filter->real_vol_f = 0.0; filter->real_vol_i = 0; } else { filter->real_vol_f = filter->volume_f; filter->real_vol_i = filter->volume_i; } } static void gst_volume_get_volume (GstMixer *mixer, GstMixerTrack *track, gint *volumes) { GstVolume *filter = GST_VOLUME (mixer); g_return_if_fail (filter != NULL); g_return_if_fail (GST_IS_VOLUME (filter)); volumes[0] = (gint) filter->volume_f * VOLUME_STEPS; } static void gst_volume_set_mute (GstMixer *mixer, GstMixerTrack *track, gboolean mute) { GstVolume *filter = GST_VOLUME (mixer); g_return_if_fail (filter != NULL); g_return_if_fail (GST_IS_VOLUME (filter)); gst_dpman_bypass_dparam (filter->dpman, "volume"); filter->mute = mute; if (filter->mute) { filter->real_vol_f = 0.0; filter->real_vol_i = 0; } else { filter->real_vol_f = filter->volume_f; filter->real_vol_i = filter->volume_i; } } static void gst_volume_mixer_init (GstMixerClass *klass) { GST_MIXER_TYPE (klass) = GST_MIXER_SOFTWARE; /* default virtual functions */ klass->list_tracks = gst_volume_list_tracks; klass->set_volume = gst_volume_set_volume; klass->get_volume = gst_volume_get_volume; klass->set_mute = gst_volume_set_mute; } static void gst_volume_dispose (GObject *object) { GstVolume *volume; volume = GST_VOLUME (object); if (volume->tracklist) { if (volume->tracklist->data) g_object_unref (volume->tracklist->data); g_list_free (volume->tracklist); volume->tracklist = NULL; } G_OBJECT_CLASS (parent_class)->dispose (object); } static GstPadLinkReturn volume_connect (GstPad *pad, const GstCaps *caps) { GstVolume *filter; GstPad *otherpad; gint rate; GstPadLinkReturn link_ret; GstStructure *structure; filter = GST_VOLUME (gst_pad_get_parent (pad)); g_return_val_if_fail (GST_IS_VOLUME (filter), GST_PAD_LINK_REFUSED); otherpad = (pad == filter->srcpad ? filter->sinkpad : filter->srcpad); structure = gst_caps_get_structure (caps, 0); gst_structure_get_int (structure, "rate", &rate); link_ret = gst_pad_try_set_caps (otherpad, caps); if (GST_PAD_LINK_FAILED (link_ret)){ return link_ret; } if (!volume_parse_caps (filter, structure)) return GST_PAD_LINK_REFUSED; gst_dpman_set_rate(filter->dpman, rate); return GST_PAD_LINK_OK; } static gboolean volume_parse_caps (GstVolume *filter, GstStructure *structure) { const gchar *mimetype; g_return_val_if_fail(filter!=NULL,FALSE); g_return_val_if_fail(structure!=NULL,FALSE); mimetype = gst_structure_get_name (structure); if (strcmp(mimetype, "audio/x-raw-int")==0) { gst_pad_set_chain_function(filter->sinkpad,volume_chain_int16); return TRUE; } if (strcmp(mimetype, "audio/x-raw-float")==0) { gst_pad_set_chain_function(filter->sinkpad,volume_chain_float); return TRUE; } return FALSE; } GType gst_volume_get_type(void) { static GType volume_type = 0; if (!volume_type) { static const GTypeInfo volume_info = { sizeof(GstVolumeClass), volume_base_init, NULL, (GClassInitFunc)volume_class_init, NULL, NULL, sizeof(GstVolume), 0, (GInstanceInitFunc)volume_init }; static const GInterfaceInfo voliface_info = { (GInterfaceInitFunc) gst_volume_interface_init, NULL, NULL }; static const GInterfaceInfo volmixer_info = { (GInterfaceInitFunc) gst_volume_mixer_init, NULL, NULL }; volume_type = g_type_register_static(GST_TYPE_ELEMENT, "GstVolume", &volume_info, 0); g_type_add_interface_static (volume_type, GST_TYPE_IMPLEMENTS_INTERFACE, &voliface_info); g_type_add_interface_static (volume_type, GST_TYPE_MIXER, &volmixer_info); } return volume_type; } static void volume_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 (&volume_src_factory)); gst_element_class_add_pad_template (element_class, gst_static_pad_template_get (&volume_sink_factory)); gst_element_class_set_details (element_class, &volume_details); } static void volume_class_init (GstVolumeClass *klass) { GObjectClass *gobject_class; GstElementClass *gstelement_class; gobject_class = (GObjectClass*)klass; gstelement_class = (GstElementClass*)klass; parent_class = g_type_class_ref(GST_TYPE_ELEMENT); g_object_class_install_property(G_OBJECT_CLASS(klass), ARG_MUTE, g_param_spec_boolean("mute","mute","mute", FALSE,G_PARAM_READWRITE)); g_object_class_install_property(G_OBJECT_CLASS(klass), ARG_VOLUME, g_param_spec_double("volume","volume","volume", 0.0,VOLUME_MAX_DOUBLE,1.0,G_PARAM_READWRITE)); gobject_class->set_property = volume_set_property; gobject_class->get_property = volume_get_property; gobject_class->dispose = gst_volume_dispose; } static void volume_init (GstVolume *filter) { GstMixerTrack *track = NULL; filter->sinkpad = gst_pad_new_from_template ( gst_static_pad_template_get (&volume_sink_factory), "sink"); gst_pad_set_getcaps_function (filter->sinkpad, gst_pad_proxy_getcaps); gst_pad_set_link_function (filter->sinkpad, volume_connect); filter->srcpad = gst_pad_new_from_template ( gst_static_pad_template_get (&volume_src_factory), "src"); gst_pad_set_getcaps_function (filter->srcpad, gst_pad_proxy_getcaps); gst_pad_set_link_function(filter->srcpad,volume_connect); gst_element_add_pad (GST_ELEMENT (filter), filter->sinkpad); gst_element_add_pad (GST_ELEMENT (filter), filter->srcpad); gst_pad_set_chain_function (filter->sinkpad, volume_chain_int16); filter->mute = FALSE; filter->volume_i = VOLUME_UNITY_INT; filter->volume_f = 1.0; filter->real_vol_i = VOLUME_UNITY_INT; filter->real_vol_f = 1.0; filter->tracklist = NULL; filter->dpman = gst_dpman_new ("volume_dpman", GST_ELEMENT(filter)); gst_dpman_add_required_dparam_callback ( filter->dpman, g_param_spec_int("mute","Mute","Mute the audio", 0, 1, 0, G_PARAM_READWRITE), "int", volume_update_mute, filter ); gst_dpman_add_required_dparam_callback ( filter->dpman, g_param_spec_double("volume","Volume","Volume of the audio", 0.0, VOLUME_MAX_DOUBLE, 1.0, G_PARAM_READWRITE), "scalar", volume_update_volume, filter ); track = g_object_new (GST_TYPE_MIXER_TRACK, NULL); if (GST_IS_MIXER_TRACK (track)) { track->label = g_strdup ("volume"); track->num_channels = 1; track->min_volume = 0; track->max_volume = VOLUME_STEPS; track->flags = GST_MIXER_TRACK_SOFTWARE; filter->tracklist = g_list_append (filter->tracklist, track); } } static void volume_chain_float (GstPad *pad, GstData *_data) { GstBuffer *buf = GST_BUFFER (_data); GstVolume *filter; GstBuffer *out_buf; gfloat *data; gint i, num_samples; g_return_if_fail(GST_IS_PAD(pad)); g_return_if_fail(buf != NULL); filter = GST_VOLUME(GST_OBJECT_PARENT (pad)); g_return_if_fail(GST_IS_VOLUME(filter)); out_buf = gst_buffer_copy_on_write (buf); data = (gfloat *)GST_BUFFER_DATA(out_buf); num_samples = GST_BUFFER_SIZE(out_buf)/sizeof(gfloat); GST_DPMAN_PREPROCESS(filter->dpman, num_samples, GST_BUFFER_TIMESTAMP(out_buf)); i = 0; while(GST_DPMAN_PROCESS(filter->dpman, i)) { data[i++] *= filter->real_vol_f; } gst_pad_push(filter->srcpad,GST_DATA (out_buf)); } static void volume_chain_int16 (GstPad *pad, GstData *_data) { GstBuffer *buf = GST_BUFFER (_data); GstVolume *filter; GstBuffer *out_buf; gint16 *data; gint i, num_samples; g_return_if_fail(GST_IS_PAD(pad)); g_return_if_fail(buf != NULL); filter = GST_VOLUME(GST_OBJECT_PARENT (pad)); g_return_if_fail(GST_IS_VOLUME(filter)); out_buf = gst_buffer_copy_on_write (buf); data = (gint16 *) GST_BUFFER_DATA (out_buf); g_assert (data); num_samples = GST_BUFFER_SIZE(out_buf)/sizeof(gint16); GST_DPMAN_PREPROCESS(filter->dpman, num_samples, GST_BUFFER_TIMESTAMP(out_buf)); i = 0; while(GST_DPMAN_PROCESS(filter->dpman, i)) { /* only clamp if the gain is greater than 1.0 */ if (filter->real_vol_i > VOLUME_UNITY_INT){ while (i < GST_DPMAN_NEXT_UPDATE_FRAME(filter->dpman)){ /* we use bitshifting instead of dividing by UNITY_INT for speed */ data[i] = (gint16) CLAMP((filter->real_vol_i * (gint) data[i]) >> VOLUME_UNITY_BIT_SHIFT, VOLUME_MIN_INT16, VOLUME_MAX_INT16); i++; } } else { while (i < GST_DPMAN_NEXT_UPDATE_FRAME(filter->dpman)){ /* we use bitshifting instead of dividing by UNITY_INT for speed */ data[i] = (gint16) ((filter->real_vol_i * (gint) data[i]) >> VOLUME_UNITY_BIT_SHIFT); i++; } } } gst_pad_push(filter->srcpad,GST_DATA (out_buf)); } static void volume_update_mute(const GValue *value, gpointer data) { GstVolume *filter = (GstVolume*)data; g_return_if_fail(GST_IS_VOLUME(filter)); if (G_VALUE_HOLDS_BOOLEAN(value)){ filter->mute = g_value_get_boolean(value); } else if (G_VALUE_HOLDS_INT(value)){ filter->mute = (g_value_get_int(value) == 1); } if (filter->mute){ filter->real_vol_f = 0.0; filter->real_vol_i = 0; } else { filter->real_vol_f = filter->volume_f; filter->real_vol_i = filter->volume_i; } } static void volume_update_volume(const GValue *value, gpointer data) { GstVolume *filter = (GstVolume*)data; g_return_if_fail(GST_IS_VOLUME(filter)); filter->volume_f = g_value_get_double (value); filter->volume_i = filter->volume_f*VOLUME_UNITY_INT; if (filter->mute){ filter->real_vol_f = 0.0; filter->real_vol_i = 0; } else { filter->real_vol_f = filter->volume_f; filter->real_vol_i = filter->volume_i; } } static void volume_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) { GstVolume *filter; /* it's not null if we got it, but it might not be ours */ g_return_if_fail(GST_IS_VOLUME(object)); filter = GST_VOLUME(object); switch (prop_id) { case ARG_MUTE: gst_dpman_bypass_dparam(filter->dpman, "mute"); volume_update_mute(value, filter); break; case ARG_VOLUME: gst_dpman_bypass_dparam(filter->dpman, "volume"); volume_update_volume(value, filter); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void volume_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { GstVolume *filter; /* it's not null if we got it, but it might not be ours */ g_return_if_fail(GST_IS_VOLUME(object)); filter = GST_VOLUME(object); switch (prop_id) { case ARG_MUTE: g_value_set_boolean (value, filter->mute); break; case ARG_VOLUME: g_value_set_double (value, filter->volume_f); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static gboolean plugin_init (GstPlugin *plugin) { gst_control_init(NULL,NULL); return gst_element_register (plugin, "volume", GST_RANK_NONE, GST_TYPE_VOLUME); } GST_PLUGIN_DEFINE ( GST_VERSION_MAJOR, GST_VERSION_MINOR, "volume", "element for controlling audio volume", plugin_init, VERSION, GST_LICENSE, GST_PACKAGE, GST_ORIGIN )