/* GStreamer Mixer * Copyright (C) 2003 Ronald Bultje * * mixer.c: mixer design virtual class function wrappers * * 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 "mixer.h" #include "interfaces-marshal.h" #define GST_MIXER_MESSAGE_NAME "gst-mixer-message" /** * SECTION:gstmixer * @short_description: Interface for elements that provide mixer operations */ #ifndef GST_DISABLE_DEPRECATED enum { SIGNAL_MUTE_TOGGLED, SIGNAL_RECORD_TOGGLED, SIGNAL_VOLUME_CHANGED, SIGNAL_OPTION_CHANGED, LAST_SIGNAL }; static guint gst_mixer_signals[LAST_SIGNAL] = { 0 }; #endif static void gst_mixer_class_init (GstMixerClass * klass); GType gst_mixer_get_type (void) { static GType gst_mixer_type = 0; if (!gst_mixer_type) { static const GTypeInfo gst_mixer_info = { sizeof (GstMixerClass), (GBaseInitFunc) gst_mixer_class_init, NULL, NULL, NULL, NULL, 0, 0, NULL, }; gst_mixer_type = g_type_register_static (G_TYPE_INTERFACE, "GstMixer", &gst_mixer_info, 0); g_type_interface_add_prerequisite (gst_mixer_type, GST_TYPE_IMPLEMENTS_INTERFACE); } return gst_mixer_type; } static void gst_mixer_class_init (GstMixerClass * klass) { #ifndef GST_DISABLE_DEPRECATED static gboolean initialized = FALSE; /* signals (deprecated) */ if (!initialized) { gst_mixer_signals[SIGNAL_RECORD_TOGGLED] = g_signal_new ("record-toggled", GST_TYPE_MIXER, G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (GstMixerClass, record_toggled), NULL, NULL, gst_interfaces_marshal_VOID__OBJECT_BOOLEAN, G_TYPE_NONE, 2, GST_TYPE_MIXER_TRACK, G_TYPE_BOOLEAN); gst_mixer_signals[SIGNAL_MUTE_TOGGLED] = g_signal_new ("mute-toggled", GST_TYPE_MIXER, G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (GstMixerClass, mute_toggled), NULL, NULL, gst_interfaces_marshal_VOID__OBJECT_BOOLEAN, G_TYPE_NONE, 2, GST_TYPE_MIXER_TRACK, G_TYPE_BOOLEAN); gst_mixer_signals[SIGNAL_VOLUME_CHANGED] = g_signal_new ("volume-changed", GST_TYPE_MIXER, G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (GstMixerClass, volume_changed), NULL, NULL, gst_interfaces_marshal_VOID__OBJECT_POINTER, G_TYPE_NONE, 2, GST_TYPE_MIXER_TRACK, G_TYPE_POINTER); gst_mixer_signals[SIGNAL_OPTION_CHANGED] = g_signal_new ("option-changed", GST_TYPE_MIXER, G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (GstMixerClass, option_changed), NULL, NULL, gst_interfaces_marshal_VOID__OBJECT_STRING, G_TYPE_NONE, 2, GST_TYPE_MIXER_OPTIONS, G_TYPE_STRING); initialized = TRUE; } #endif klass->mixer_type = GST_MIXER_SOFTWARE; /* default virtual functions */ klass->list_tracks = NULL; klass->set_volume = NULL; klass->get_volume = NULL; klass->set_mute = NULL; klass->set_record = NULL; klass->set_option = NULL; klass->get_option = NULL; } /** * gst_mixer_list_tracks: * @mixer: the #GstMixer (a #GstElement) to get the tracks from. * * Returns a list of available tracks for this mixer/element. Note * that it is allowed for sink (output) elements to only provide * the output tracks in this list. Likewise, for sources (inputs), * it is allowed to only provide input elements in this list. * * Returns: A #GList consisting of zero or more #GstMixerTracks. */ const GList * gst_mixer_list_tracks (GstMixer * mixer) { GstMixerClass *klass; g_return_val_if_fail (mixer != NULL, NULL); klass = GST_MIXER_GET_CLASS (mixer); if (klass->list_tracks) { return klass->list_tracks (mixer); } return NULL; } /** * gst_mixer_set_volume: * @mixer: The #GstMixer (a #GstElement) that owns the track. * @track: The #GstMixerTrack to set the volume on. * @volumes: an array of integers (of size track->num_channels) * that gives the wanted volume for each channel in * this track. * * Sets the volume on each channel in a track. Short note about * naming: a track is defined as one separate stream owned by * the mixer/element, such as 'Line-in' or 'Microphone'. A * channel is said to be a mono-stream inside this track. A * stereo track thus contains two channels. */ void gst_mixer_set_volume (GstMixer * mixer, GstMixerTrack * track, gint * volumes) { GstMixerClass *klass; g_return_if_fail (mixer != NULL); g_return_if_fail (track != NULL); g_return_if_fail (volumes != NULL); klass = GST_MIXER_GET_CLASS (mixer); if (klass->set_volume) { klass->set_volume (mixer, track, volumes); } } /** * gst_mixer_get_volume: * @mixer: the #GstMixer (a #GstElement) that owns the track * @track: the GstMixerTrack to get the volume from. * @volumes: a pre-allocated array of integers (of size * track->num_channels) to store the current volume * of each channel in the given track in. * * Get the current volume(s) on the given track. */ void gst_mixer_get_volume (GstMixer * mixer, GstMixerTrack * track, gint * volumes) { GstMixerClass *klass; g_return_if_fail (mixer != NULL); g_return_if_fail (track != NULL); g_return_if_fail (volumes != NULL); klass = GST_MIXER_GET_CLASS (mixer); if (klass->get_volume) { klass->get_volume (mixer, track, volumes); } else { gint i; for (i = 0; i < track->num_channels; i++) { volumes[i] = 0; } } } /** * gst_mixer_set_mute: * @mixer: the #GstMixer (a #GstElement) that owns the track. * @track: the #GstMixerTrack to operate on. * @mute: a boolean value indicating whether to turn on or off * muting. * * Mutes or unmutes the given channel. To find out whether a * track is currently muted, use GST_MIXER_TRACK_HAS_FLAG (). */ void gst_mixer_set_mute (GstMixer * mixer, GstMixerTrack * track, gboolean mute) { GstMixerClass *klass; g_return_if_fail (mixer != NULL); g_return_if_fail (track != NULL); klass = GST_MIXER_GET_CLASS (mixer); if (klass->set_mute) { klass->set_mute (mixer, track, mute); } } /** * gst_mixer_set_record: * @mixer: The #GstMixer (a #GstElement) that owns the track. * @track: the #GstMixerTrack to operate on. * @record: a boolean value that indicates whether to turn on * or off recording. * * Enables or disables recording on the given track. Note that * this is only possible on input tracks, not on output tracks * (see GST_MIXER_TRACK_HAS_FLAG () and the GST_MIXER_TRACK_INPUT * flag). */ void gst_mixer_set_record (GstMixer * mixer, GstMixerTrack * track, gboolean record) { GstMixerClass *klass = GST_MIXER_GET_CLASS (mixer); if (klass->set_record) { klass->set_record (mixer, track, record); } } /** * gst_mixer_set_option: * @mixer: The #GstMixer (a #GstElement) that owns the optionlist. * @opts: The #GstMixerOptions that we operate on. * @value: The requested new option value. * * Sets a name/value option in the mixer to the requested value. */ void gst_mixer_set_option (GstMixer * mixer, GstMixerOptions * opts, gchar * value) { GstMixerClass *klass; g_return_if_fail (mixer != NULL); g_return_if_fail (opts != NULL); klass = GST_MIXER_GET_CLASS (mixer); if (klass->set_option) { klass->set_option (mixer, opts, value); } } /** * gst_mixer_get_option: * @mixer: The #GstMixer (a #GstElement) that owns the optionlist. * @opts: The #GstMixerOptions that we operate on. * * Get the current value of a name/value option in the mixer. * * Returns: current value of the name/value option. */ const gchar * gst_mixer_get_option (GstMixer * mixer, GstMixerOptions * opts) { GstMixerClass *klass; g_return_val_if_fail (mixer != NULL, NULL); g_return_val_if_fail (opts != NULL, NULL); klass = GST_MIXER_GET_CLASS (mixer); if (klass->get_option) { return klass->get_option (mixer, opts); } return NULL; } /** * gst_mixer_get_mixer_flags: * @mixer: The #GstMixer implementation * * Get the set of supported flags for this mixer implementation. * * Returns: A set of or-ed GstMixerFlags for supported features. */ GstMixerFlags gst_mixer_get_mixer_flags (GstMixer * mixer) { GstMixerClass *klass; g_return_val_if_fail (mixer != NULL, FALSE); klass = GST_MIXER_GET_CLASS (mixer); if (klass->get_mixer_flags) { return klass->get_mixer_flags (mixer); } return GST_MIXER_FLAG_NONE; } /** * gst_mixer_mute_toggled: * @mixer: the #GstMixer (a #GstElement) that owns the track * @track: the GstMixerTrack that has change mute state. * @mute: the new state of the mute flag on the track * * This function is called by the mixer implementation to produce * a notification message on the bus indicating that the given track * has changed mute state. * * This function only works for GstElements that are implementing the * GstMixer interface, and the element needs to have been provided a bus. */ void gst_mixer_mute_toggled (GstMixer * mixer, GstMixerTrack * track, gboolean mute) { GstStructure *s; GstMessage *m; g_return_if_fail (mixer != NULL); g_return_if_fail (GST_IS_ELEMENT (mixer)); g_return_if_fail (track != NULL); s = gst_structure_new (GST_MIXER_MESSAGE_NAME, "type", G_TYPE_STRING, "mute-toggled", "track", GST_TYPE_MIXER_TRACK, track, "mute", G_TYPE_BOOLEAN, mute, NULL); m = gst_message_new_element (GST_OBJECT (mixer), s); if (gst_element_post_message (GST_ELEMENT (mixer), m) == FALSE) { GST_WARNING ("This element has no bus, therefore no message sent!"); } } /** * gst_mixer_record_toggled: * @mixer: the #GstMixer (a #GstElement) that owns the track * @track: the GstMixerTrack that has changed recording state. * @record: the new state of the record flag on the track * * This function is called by the mixer implementation to produce * a notification message on the bus indicating that the given track * has changed recording state. * * This function only works for GstElements that are implementing the * GstMixer interface, and the element needs to have been provided a bus. */ void gst_mixer_record_toggled (GstMixer * mixer, GstMixerTrack * track, gboolean record) { GstStructure *s; GstMessage *m; g_return_if_fail (mixer != NULL); g_return_if_fail (GST_IS_ELEMENT (mixer)); g_return_if_fail (track != NULL); s = gst_structure_new (GST_MIXER_MESSAGE_NAME, "type", G_TYPE_STRING, "record-toggled", "track", GST_TYPE_MIXER_TRACK, track, "record", G_TYPE_BOOLEAN, record, NULL); m = gst_message_new_element (GST_OBJECT (mixer), s); if (gst_element_post_message (GST_ELEMENT (mixer), m) == FALSE) { GST_WARNING ("This element has no bus, therefore no message sent!"); } } /** * gst_mixer_volume_changed: * @mixer: the #GstMixer (a #GstElement) that owns the track * @track: the GstMixerTrack that has changed. * @volumes: Array of volume values, one per channel on the mixer track. * * This function is called by the mixer implementation to produce * a notification message on the bus indicating that the volume(s) for the * given track have changed. * * This function only works for GstElements that are implementing the * GstMixer interface, and the element needs to have been provided a bus. */ void gst_mixer_volume_changed (GstMixer * mixer, GstMixerTrack * track, gint * volumes) { GstStructure *s; GstMessage *m; GValue l = { 0, }; GValue v = { 0, }; gint i; g_return_if_fail (mixer != NULL); g_return_if_fail (GST_IS_ELEMENT (mixer)); g_return_if_fail (track != NULL); s = gst_structure_new (GST_MIXER_MESSAGE_NAME, "type", G_TYPE_STRING, "volume-changed", "track", GST_TYPE_MIXER_TRACK, track, NULL); g_value_init (&l, GST_TYPE_ARRAY); g_value_init (&v, G_TYPE_INT); /* FIXME 0.11: pass track->num_channels to the function */ for (i = 0; i < track->num_channels; ++i) { g_value_set_int (&v, volumes[i]); gst_value_array_append_value (&l, &v); } g_value_unset (&v); gst_structure_set_value (s, "volumes", &l); g_value_unset (&l); m = gst_message_new_element (GST_OBJECT (mixer), s); if (gst_element_post_message (GST_ELEMENT (mixer), m) == FALSE) { GST_WARNING ("This element has no bus, therefore no message sent!"); } } /** * gst_mixer_option_changed: * @mixer: the #GstMixer (a #GstElement) that owns the options * @opts: the GstMixerOptions that has changed value. * @value: the new value of the GstMixerOptions. * * This function is called by the mixer implementation to produce * a notification message on the bus indicating that the given options * object has changed state. * * This function only works for GstElements that are implementing the * GstMixer interface, and the element needs to have been provided a bus. */ void gst_mixer_option_changed (GstMixer * mixer, GstMixerOptions * opts, gchar * value) { GstStructure *s; GstMessage *m; g_return_if_fail (mixer != NULL); g_return_if_fail (GST_IS_ELEMENT (mixer)); g_return_if_fail (opts != NULL); s = gst_structure_new (GST_MIXER_MESSAGE_NAME, "type", G_TYPE_STRING, "option-changed", "options", GST_TYPE_MIXER_OPTIONS, opts, "value", G_TYPE_STRING, value, NULL); m = gst_message_new_element (GST_OBJECT (mixer), s); if (gst_element_post_message (GST_ELEMENT (mixer), m) == FALSE) { GST_WARNING ("This element has no bus, therefore no message sent!"); } } static gboolean gst_mixer_message_is_mixer_message (GstMessage * message) { const GstStructure *s; if (message == NULL) return FALSE; if (GST_MESSAGE_TYPE (message) != GST_MESSAGE_ELEMENT) return FALSE; s = gst_message_get_structure (message); return gst_structure_has_name (s, GST_MIXER_MESSAGE_NAME); } /** * gst_mixer_message_get_type: * @message: A GstMessage to inspect. * * Check a bus message to see if it is a GstMixer notification * message and return the GstMixerMessageType identifying which * type of notification it is. * * Returns: The type of the GstMixerMessage, or GST_MIXER_MESSAGE_NONE * if the message is not a GstMixer notification. * * Since: 0.10.14 */ GstMixerMessageType gst_mixer_message_get_type (GstMessage * message) { const GstStructure *s; const gchar *m_type; if (!gst_mixer_message_is_mixer_message (message)) return GST_MIXER_MESSAGE_INVALID; s = gst_message_get_structure (message); m_type = gst_structure_get_string (s, "type"); g_return_val_if_fail (m_type != NULL, GST_MIXER_MESSAGE_INVALID); if (g_str_equal (m_type, "mute-toggled")) return GST_MIXER_MESSAGE_MUTE_TOGGLED; else if (g_str_equal (m_type, "record-toggled")) return GST_MIXER_MESSAGE_RECORD_TOGGLED; else if (g_str_equal (m_type, "volume-changed")) return GST_MIXER_MESSAGE_VOLUME_CHANGED; else if (g_str_equal (m_type, "option-changed")) return GST_MIXER_MESSAGE_OPTION_CHANGED; return GST_MIXER_MESSAGE_INVALID; } #define GST_MIXER_MESSAGE_HAS_TYPE(msg,msg_type) \ (gst_mixer_message_get_type (msg) == GST_MIXER_MESSAGE_ ## msg_type) /** * gst_mixer_message_parse_mute_toggled: * @message: A mute-toggled change notification message. * @track: Pointer to hold a GstMixerTrack object, or NULL. * @mute: A pointer to a gboolean variable, or NULL. * * Extracts the contents of a mute-toggled bus message. Reads * the GstMixerTrack that has changed, and the new value of the mute * flag. * * The GstMixerTrack remains valid until the message is freed. * * Since: 0.10.14 */ void gst_mixer_message_parse_mute_toggled (GstMessage * message, GstMixerTrack ** track, gboolean * mute) { const GstStructure *s; g_return_if_fail (gst_mixer_message_is_mixer_message (message)); g_return_if_fail (GST_MIXER_MESSAGE_HAS_TYPE (message, MUTE_TOGGLED)); s = gst_message_get_structure (message); if (track) { const GValue *v = gst_structure_get_value (s, "track"); g_return_if_fail (v != NULL); *track = (GstMixerTrack *) g_value_get_object (v); g_return_if_fail (GST_IS_MIXER_TRACK (*track)); } if (mute) g_return_if_fail (gst_structure_get_boolean (s, "mute", mute)); } /** * gst_mixer_message_parse_record_toggled: * @message: A record-toggled change notification message. * @track: Pointer to hold a GstMixerTrack object, or NULL. * @record: A pointer to a gboolean variable, or NULL. * * Extracts the contents of a record-toggled bus message. Reads * the GstMixerTrack that has changed, and the new value of the * recording flag. * * The GstMixerTrack remains valid until the message is freed. * * Since: 0.10.14 */ void gst_mixer_message_parse_record_toggled (GstMessage * message, GstMixerTrack ** track, gboolean * record) { const GstStructure *s; g_return_if_fail (gst_mixer_message_is_mixer_message (message)); g_return_if_fail (GST_MIXER_MESSAGE_HAS_TYPE (message, RECORD_TOGGLED)); s = gst_message_get_structure (message); if (track) { const GValue *v = gst_structure_get_value (s, "track"); g_return_if_fail (v != NULL); *track = (GstMixerTrack *) g_value_get_object (v); g_return_if_fail (GST_IS_MIXER_TRACK (*track)); } if (record) g_return_if_fail (gst_structure_get_boolean (s, "record", record)); } /** * gst_mixer_message_parse_volume_changed: * @message: A volume-changed change notification message. * @track: Pointer to hold a GstMixerTrack object, or NULL. * @volumes: A pointer to receive an array of gint values, or NULL. * @num_channels: Result location to receive the number of channels, or NULL. * * Parses a volume-changed notification message and extracts the track object * it refers to, as well as an array of volumes and the size of the volumes array. * * The track object remains valid until the message is freed. * * The caller must free the array returned in the volumes parameter using g_free * when they are done with it. * * Since: 0.10.14 */ void gst_mixer_message_parse_volume_changed (GstMessage * message, GstMixerTrack ** track, gint ** volumes, gint * num_channels) { const GstStructure *s; g_return_if_fail (gst_mixer_message_is_mixer_message (message)); g_return_if_fail (GST_MIXER_MESSAGE_HAS_TYPE (message, VOLUME_CHANGED)); s = gst_message_get_structure (message); if (track) { const GValue *v = gst_structure_get_value (s, "track"); g_return_if_fail (v != NULL); *track = (GstMixerTrack *) g_value_get_object (v); g_return_if_fail (GST_IS_MIXER_TRACK (*track)); } if (volumes || num_channels) { gint n_chans, i; const GValue *v = gst_structure_get_value (s, "volumes"); g_return_if_fail (v != NULL); g_return_if_fail (GST_VALUE_HOLDS_ARRAY (v)); n_chans = gst_value_array_get_size (v); if (num_channels) *num_channels = n_chans; if (volumes) { *volumes = g_new (gint, n_chans); for (i = 0; i < n_chans; i++) { const GValue *e = gst_value_array_get_value (v, i); g_return_if_fail (e != NULL && G_VALUE_HOLDS_INT (e)); (*volumes)[i] = g_value_get_int (e); } } } } /** * gst_mixer_message_parse_option_changed: * @message: A volume-changed change notification message. * @options: Pointer to hold a GstMixerOptions object, or NULL. * @value: Result location to receive the new options value, or NULL. * * Extracts the GstMixerOptions and new value from a option-changed bus notification * message. * * The options and value returned remain valid until the message is freed. * * Since: 0.10.14 */ void gst_mixer_message_parse_option_changed (GstMessage * message, GstMixerOptions ** options, const gchar ** value) { const GstStructure *s; g_return_if_fail (gst_mixer_message_is_mixer_message (message)); g_return_if_fail (GST_MIXER_MESSAGE_HAS_TYPE (message, OPTION_CHANGED)); s = gst_message_get_structure (message); if (options) { const GValue *v = gst_structure_get_value (s, "options"); g_return_if_fail (v != NULL); *options = (GstMixerOptions *) g_value_get_object (v); g_return_if_fail (GST_IS_MIXER_OPTIONS (*options)); } if (value) *value = gst_structure_get_string (s, "value"); }