mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-16 12:25:50 +00:00
a788c9ae61
Original commit message from CVS: Reviewed by: Ronald S. Bultje <rbultje@ronald.bitfreak.net> * sys/sunaudio/gstsunmixer.c: (gst_sunaudiomixer_set_volume): Normalizing the value before setting (gst_sunaudiomixer_get_volume): Normalizing the value after getting. Fixes bug# 161980
306 lines
7.8 KiB
C
306 lines
7.8 KiB
C
/*
|
|
* GStreamer
|
|
* Copyright (C) 1999-2001 Erik Walthinsen <omega@cse.ogi.edu>
|
|
* Copyright (C) 2004 David A. Schleef <ds@schleef.org>
|
|
*
|
|
* 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 <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
#include <sys/ioctl.h>
|
|
#include <sys/audioio.h>
|
|
|
|
#include "gst/gst-i18n-plugin.h"
|
|
#include "gstsunmixer.h"
|
|
|
|
static void gst_sunaudiomixer_track_class_init (GstSunAudioMixerTrackClass *
|
|
klass);
|
|
static void gst_sunaudiomixer_track_init (GstSunAudioMixerTrack * track);
|
|
|
|
static gboolean gst_sunaudiomixer_supported (GstImplementsInterface * iface,
|
|
GType iface_type);
|
|
static const GList *gst_sunaudiomixer_list_tracks (GstMixer * sunaudiomixer);
|
|
static void gst_sunaudiomixer_set_volume (GstMixer * sunaudiomixer,
|
|
GstMixerTrack * track, gint * volumes);
|
|
static void gst_sunaudiomixer_get_volume (GstMixer * sunaudiomixer,
|
|
GstMixerTrack * track, gint * volumes);
|
|
static void gst_sunaudiomixer_set_mute (GstMixer * sunaudiomixer,
|
|
GstMixerTrack * track, gboolean mute);
|
|
static void gst_sunaudiomixer_set_record (GstMixer * sunaudiomixer,
|
|
GstMixerTrack * track, gboolean record);
|
|
|
|
#define MIXER_DEVICES 3
|
|
#define SCALE_FACTOR 2.55 /* 255/100 */
|
|
static gchar **labels = NULL;
|
|
static GstMixerTrackClass *parent_class = NULL;
|
|
|
|
GType
|
|
gst_sunaudiomixer_track_get_type (void)
|
|
{
|
|
static GType gst_sunaudiomixer_track_type = 0;
|
|
|
|
if (!gst_sunaudiomixer_track_type) {
|
|
static const GTypeInfo sunaudiomixer_track_info = {
|
|
sizeof (GstSunAudioMixerTrackClass),
|
|
NULL,
|
|
NULL,
|
|
(GClassInitFunc) gst_sunaudiomixer_track_class_init,
|
|
NULL,
|
|
NULL,
|
|
sizeof (GstSunAudioMixerTrack),
|
|
0,
|
|
(GInstanceInitFunc) gst_sunaudiomixer_track_init,
|
|
NULL
|
|
};
|
|
|
|
gst_sunaudiomixer_track_type = g_type_register_static (GST_TYPE_MIXER_TRACK,
|
|
"GstSunAudioMixerTrack", &sunaudiomixer_track_info, 0);
|
|
|
|
}
|
|
|
|
return gst_sunaudiomixer_track_type;
|
|
}
|
|
|
|
static void
|
|
gst_sunaudiomixer_track_class_init (GstSunAudioMixerTrackClass * klass)
|
|
{
|
|
parent_class = g_type_class_ref (GST_TYPE_MIXER_TRACK);
|
|
}
|
|
|
|
static void
|
|
gst_sunaudiomixer_track_init (GstSunAudioMixerTrack * track)
|
|
{
|
|
track->lvol = track->rvol = 0;
|
|
track->track_num = 0;
|
|
}
|
|
|
|
void
|
|
gst_sunaudio_interface_init (GstImplementsInterfaceClass * klass)
|
|
{
|
|
/* default virtual functions */
|
|
klass->supported = gst_sunaudiomixer_supported;
|
|
}
|
|
|
|
static gboolean
|
|
gst_sunaudiomixer_supported (GstImplementsInterface * iface, GType iface_type)
|
|
{
|
|
g_assert (iface_type == GST_TYPE_MIXER);
|
|
|
|
return (GST_SUNAUDIOELEMENT (iface)->mixer_fd != -1);
|
|
}
|
|
|
|
void
|
|
gst_sunaudiomixer_interface_init (GstMixerClass * klass)
|
|
{
|
|
GST_MIXER_TYPE (klass) = GST_MIXER_HARDWARE;
|
|
|
|
klass->list_tracks = gst_sunaudiomixer_list_tracks;
|
|
klass->set_volume = gst_sunaudiomixer_set_volume;
|
|
klass->get_volume = gst_sunaudiomixer_get_volume;
|
|
klass->set_mute = gst_sunaudiomixer_set_mute;
|
|
klass->set_record = gst_sunaudiomixer_set_record;
|
|
}
|
|
|
|
static void
|
|
fill_labels (void)
|
|
{
|
|
int i;
|
|
struct
|
|
{
|
|
gchar *given, *wanted;
|
|
}
|
|
cases[] = {
|
|
{
|
|
"Vol ", N_("Volume")}
|
|
, {
|
|
"Gain ", N_("Gain")}
|
|
, {
|
|
"Mon ", N_("Monitor")}
|
|
, {
|
|
NULL, NULL}
|
|
};
|
|
|
|
labels = g_malloc (sizeof (gchar *) * MIXER_DEVICES);
|
|
|
|
for (i = 0; i < MIXER_DEVICES; i++) {
|
|
labels[i] = g_strdup (cases[i].wanted);
|
|
}
|
|
}
|
|
|
|
GstMixerTrack *
|
|
gst_sunaudiomixer_track_new (GstSunAudioElement * sunaudio,
|
|
gint track_num, gint max_chans, gint flags)
|
|
{
|
|
GstSunAudioMixerTrack *sunaudiotrack;
|
|
GstMixerTrack *track;
|
|
gint volume;
|
|
|
|
if (!labels)
|
|
fill_labels ();
|
|
|
|
sunaudiotrack = g_object_new (GST_TYPE_SUNAUDIOMIXER_TRACK, NULL);
|
|
track = GST_MIXER_TRACK (sunaudiotrack);
|
|
track->label = g_strdup (labels[track_num]);
|
|
track->num_channels = max_chans;
|
|
track->flags = flags;
|
|
track->min_volume = 0;
|
|
track->max_volume = 100;
|
|
sunaudiotrack->track_num = track_num;
|
|
|
|
sunaudiotrack->lvol = (0 & 0xff);
|
|
|
|
return track;
|
|
}
|
|
|
|
void
|
|
gst_sunaudiomixer_build_list (GstSunAudioElement * sunaudio)
|
|
{
|
|
GstMixerTrack *track;
|
|
|
|
sunaudio->mixer_fd = open (sunaudio->mixer_dev, O_RDWR);
|
|
|
|
if (sunaudio->mixer_fd == -1) {
|
|
return;
|
|
}
|
|
|
|
sunaudio->device_name = g_strdup ("Unknown");
|
|
|
|
track = gst_sunaudiomixer_track_new (sunaudio, 0, 1, GST_MIXER_TRACK_OUTPUT);
|
|
sunaudio->tracklist = g_list_append (sunaudio->tracklist, track);
|
|
track = gst_sunaudiomixer_track_new (sunaudio, 1, 1, 0);
|
|
sunaudio->tracklist = g_list_append (sunaudio->tracklist, track);
|
|
track = gst_sunaudiomixer_track_new (sunaudio, 2, 1, 0);
|
|
sunaudio->tracklist = g_list_append (sunaudio->tracklist, track);
|
|
}
|
|
|
|
static const GList *
|
|
gst_sunaudiomixer_list_tracks (GstMixer * mixer)
|
|
{
|
|
return (const GList *) GST_SUNAUDIOELEMENT (mixer)->tracklist;
|
|
}
|
|
|
|
static void
|
|
gst_sunaudiomixer_set_volume (GstMixer * mixer,
|
|
GstMixerTrack * track, gint * volumes)
|
|
{
|
|
gint volume;
|
|
gchar buf[100];
|
|
struct audio_info audioinfo;
|
|
GstSunAudioElement *sunaudio = GST_SUNAUDIOELEMENT (mixer);
|
|
GstSunAudioMixerTrack *sunaudiotrack = GST_SUNAUDIOMIXER_TRACK (track);
|
|
|
|
g_return_if_fail (sunaudio->mixer_fd != -1);
|
|
|
|
volume = volumes[0] * SCALE_FACTOR + 0.5;
|
|
|
|
/* Set the volume */
|
|
AUDIO_INITINFO (&audioinfo);
|
|
|
|
switch (sunaudiotrack->track_num) {
|
|
case 0:
|
|
audioinfo.play.gain = volume;
|
|
break;
|
|
case 1:
|
|
audioinfo.record.gain = volume;
|
|
break;
|
|
case 2:
|
|
audioinfo.monitor_gain = volume;
|
|
break;
|
|
}
|
|
|
|
|
|
if (ioctl (sunaudio->mixer_fd, AUDIO_SETINFO, &audioinfo) < 0) {
|
|
g_warning ("Error setting volume");
|
|
return;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_sunaudiomixer_get_volume (GstMixer * mixer,
|
|
GstMixerTrack * track, gint * volumes)
|
|
{
|
|
gint volume;
|
|
struct audio_info audioinfo;
|
|
GstSunAudioElement *sunaudio = GST_SUNAUDIOELEMENT (mixer);
|
|
GstSunAudioMixerTrack *sunaudiotrack = GST_SUNAUDIOMIXER_TRACK (track);
|
|
|
|
g_return_if_fail (sunaudio->mixer_fd != -1);
|
|
|
|
if (ioctl (sunaudio->mixer_fd, AUDIO_GETINFO, &audioinfo) < 0) {
|
|
g_warning ("Error setting volume device");
|
|
return;
|
|
}
|
|
|
|
switch (sunaudiotrack->track_num) {
|
|
case 0:
|
|
sunaudiotrack->lvol = volumes[0] =
|
|
(audioinfo.play.gain / SCALE_FACTOR) + 0.5;
|
|
break;
|
|
case 1:
|
|
sunaudiotrack->lvol = volumes[0] =
|
|
(audioinfo.record.gain / SCALE_FACTOR) + 0.5;
|
|
break;
|
|
case 2:
|
|
sunaudiotrack->lvol = volumes[0] =
|
|
(audioinfo.monitor_gain / SCALE_FACTOR) + 0.5;
|
|
break;
|
|
}
|
|
|
|
}
|
|
|
|
static void
|
|
gst_sunaudiomixer_set_mute (GstMixer * mixer,
|
|
GstMixerTrack * track, gboolean mute)
|
|
{
|
|
struct audio_info audioinfo;
|
|
GstSunAudioElement *sunaudio = GST_SUNAUDIOELEMENT (mixer);
|
|
GstSunAudioMixerTrack *sunaudiotrack = GST_SUNAUDIOMIXER_TRACK (track);
|
|
|
|
g_return_if_fail (sunaudio->mixer_fd != -1);
|
|
if (sunaudiotrack->track_num != 0)
|
|
return;
|
|
|
|
AUDIO_INITINFO (&audioinfo);
|
|
|
|
if (mute) {
|
|
audioinfo.output_muted = 1;
|
|
} else {
|
|
audioinfo.output_muted = 0;
|
|
}
|
|
|
|
if (ioctl (sunaudio->mixer_fd, AUDIO_SETINFO, &audioinfo) < 0) {
|
|
g_warning ("Error setting volume device");
|
|
return;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_sunaudiomixer_set_record (GstMixer * mixer,
|
|
GstMixerTrack * track, gboolean record)
|
|
{
|
|
|
|
/* Implementation Pending */
|
|
|
|
}
|