2 * GStreamer - SunAudio mixer interface element
3 * Copyright (C) 2005,2006,2008,2009 Sun Microsystems, Inc.,
4 * Brian Cameron <brian.cameron@sun.com>
5 * Copyright (C) 2008 Sun Microsystems, Inc.,
6 * Jan Schmidt <jan.schmidt@sun.com>
7 * Copyright (C) 2009 Sun Microsystems, Inc.,
8 * Garrett D'Amore <garrett.damore@sun.com>
10 * This library is free software; you can redistribute it and/or
11 * modify it under the terms of the GNU Library General Public
12 * License as published by the Free Software Foundation; either
13 * version 2 of the License, or (at your option) any later version.
15 * This library is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
18 * Library General Public License for more details.
20 * You should have received a copy of the GNU Library General Public
21 * License along with this library; if not, write to the
22 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
23 * Boston, MA 02111-1307, USA.
36 #include <sys/ioctl.h>
37 #include <sys/audio.h>
38 #include <sys/mixer.h>
40 #include <gst/gst-i18n-plugin.h>
42 #include "gstsunaudiomixerctrl.h"
43 #include "gstsunaudiomixertrack.h"
44 #include "gstsunaudiomixeroptions.h"
46 GST_DEBUG_CATEGORY_EXTERN (sunaudio_debug);
47 #define GST_CAT_DEFAULT sunaudio_debug
50 gst_sunaudiomixer_ctrl_open (GstSunAudioMixerCtrl * mixer)
54 /* First try to open non-blocking */
55 fd = open (mixer->device, O_RDWR | O_NONBLOCK);
59 fd = open (mixer->device, O_WRONLY);
63 GST_DEBUG_OBJECT (mixer,
64 "Failed to open mixer device %s, mixing disabled: %s", mixer->device,
71 /* Try to set the multiple open flag if we can, but ignore errors */
72 ioctl (mixer->mixer_fd, AUDIO_MIXER_MULTIPLE_OPEN);
74 GST_DEBUG_OBJECT (mixer, "Opened mixer device %s", mixer->device);
80 gst_sunaudiomixer_ctrl_build_list (GstSunAudioMixerCtrl * mixer)
83 GstMixerOptions *options;
85 struct audio_info audioinfo;
88 * Do not continue appending the same 3 static tracks onto the list
90 if (mixer->tracklist == NULL) {
91 g_return_if_fail (mixer->mixer_fd != -1);
93 /* query available ports */
94 if (ioctl (mixer->mixer_fd, AUDIO_GETINFO, &audioinfo) < 0) {
95 g_warning ("Error getting audio device volume");
99 /* Output & should be MASTER when it's the only one. */
100 track = gst_sunaudiomixer_track_new (GST_SUNAUDIO_TRACK_OUTPUT);
101 mixer->tracklist = g_list_append (mixer->tracklist, track);
104 track = gst_sunaudiomixer_track_new (GST_SUNAUDIO_TRACK_RECORD);
105 mixer->tracklist = g_list_append (mixer->tracklist, track);
108 track = gst_sunaudiomixer_track_new (GST_SUNAUDIO_TRACK_MONITOR);
109 mixer->tracklist = g_list_append (mixer->tracklist, track);
111 if (audioinfo.play.avail_ports & AUDIO_SPEAKER) {
112 track = gst_sunaudiomixer_track_new (GST_SUNAUDIO_TRACK_SPEAKER);
113 mixer->tracklist = g_list_append (mixer->tracklist, track);
115 if (audioinfo.play.avail_ports & AUDIO_HEADPHONE) {
116 track = gst_sunaudiomixer_track_new (GST_SUNAUDIO_TRACK_HP);
117 mixer->tracklist = g_list_append (mixer->tracklist, track);
119 if (audioinfo.play.avail_ports & AUDIO_LINE_OUT) {
120 track = gst_sunaudiomixer_track_new (GST_SUNAUDIO_TRACK_LINEOUT);
121 mixer->tracklist = g_list_append (mixer->tracklist, track);
123 if (audioinfo.play.avail_ports & AUDIO_SPDIF_OUT) {
124 track = gst_sunaudiomixer_track_new (GST_SUNAUDIO_TRACK_SPDIFOUT);
125 mixer->tracklist = g_list_append (mixer->tracklist, track);
127 if (audioinfo.play.avail_ports & AUDIO_AUX1_OUT) {
128 track = gst_sunaudiomixer_track_new (GST_SUNAUDIO_TRACK_AUX1OUT);
129 mixer->tracklist = g_list_append (mixer->tracklist, track);
131 if (audioinfo.play.avail_ports & AUDIO_AUX2_OUT) {
132 track = gst_sunaudiomixer_track_new (GST_SUNAUDIO_TRACK_AUX2OUT);
133 mixer->tracklist = g_list_append (mixer->tracklist, track);
136 if (audioinfo.record.avail_ports != AUDIO_NONE) {
138 gst_sunaudiomixer_options_new (mixer, GST_SUNAUDIO_TRACK_RECSRC);
139 mixer->tracklist = g_list_append (mixer->tracklist, options);
144 GstSunAudioMixerCtrl *
145 gst_sunaudiomixer_ctrl_new (const char *device)
147 GstSunAudioMixerCtrl *ret = NULL;
149 g_return_val_if_fail (device != NULL, NULL);
151 ret = g_new0 (GstSunAudioMixerCtrl, 1);
153 ret->device = g_strdup (device);
155 ret->tracklist = NULL;
157 if (!gst_sunaudiomixer_ctrl_open (ret))
164 gst_sunaudiomixer_ctrl_free (ret);
170 gst_sunaudiomixer_ctrl_free (GstSunAudioMixerCtrl * mixer)
172 g_return_if_fail (mixer != NULL);
175 g_free (mixer->device);
176 mixer->device = NULL;
179 if (mixer->tracklist) {
180 g_list_foreach (mixer->tracklist, (GFunc) g_object_unref, NULL);
181 g_list_free (mixer->tracklist);
182 mixer->tracklist = NULL;
185 if (mixer->mixer_fd != -1) {
186 close (mixer->mixer_fd);
187 mixer->mixer_fd = -1;
194 gst_sunaudiomixer_ctrl_get_mixer_flags (GstSunAudioMixerCtrl * mixer)
196 return GST_MIXER_FLAG_HAS_WHITELIST | GST_MIXER_FLAG_GROUPING;
200 gst_sunaudiomixer_ctrl_list_tracks (GstSunAudioMixerCtrl * mixer)
202 gst_sunaudiomixer_ctrl_build_list (mixer);
204 return (const GList *) mixer->tracklist;
208 gst_sunaudiomixer_ctrl_get_volume (GstSunAudioMixerCtrl * mixer,
209 GstMixerTrack * track, gint * volumes)
213 struct audio_info audioinfo;
214 GstSunAudioMixerTrack *sunaudiotrack;
216 g_return_if_fail (GST_IS_SUNAUDIO_MIXER_TRACK (track));
217 sunaudiotrack = GST_SUNAUDIO_MIXER_TRACK (track);
219 g_return_if_fail (mixer->mixer_fd != -1);
221 if (ioctl (mixer->mixer_fd, AUDIO_GETINFO, &audioinfo) < 0) {
222 g_warning ("Error getting audio device volume");
226 balance = AUDIO_MID_BALANCE;
229 switch (sunaudiotrack->track_num) {
230 case GST_SUNAUDIO_TRACK_OUTPUT:
231 gain = (int) audioinfo.play.gain;
232 balance = audioinfo.play.balance;
234 case GST_SUNAUDIO_TRACK_RECORD:
235 gain = (int) audioinfo.record.gain;
236 balance = audioinfo.record.balance;
238 case GST_SUNAUDIO_TRACK_MONITOR:
239 gain = (int) audioinfo.monitor_gain;
240 balance = audioinfo.record.balance;
242 case GST_SUNAUDIO_TRACK_SPEAKER:
243 if (audioinfo.play.port & AUDIO_SPEAKER)
244 gain = AUDIO_MAX_GAIN;
246 case GST_SUNAUDIO_TRACK_HP:
247 if (audioinfo.play.port & AUDIO_HEADPHONE)
248 gain = AUDIO_MAX_GAIN;
250 case GST_SUNAUDIO_TRACK_LINEOUT:
251 if (audioinfo.play.port & AUDIO_LINE_OUT)
252 gain = AUDIO_MAX_GAIN;
254 case GST_SUNAUDIO_TRACK_SPDIFOUT:
255 if (audioinfo.play.port & AUDIO_SPDIF_OUT)
256 gain = AUDIO_MAX_GAIN;
258 case GST_SUNAUDIO_TRACK_AUX1OUT:
259 if (audioinfo.play.port & AUDIO_AUX1_OUT)
260 gain = AUDIO_MAX_GAIN;
262 case GST_SUNAUDIO_TRACK_AUX2OUT:
263 if (audioinfo.play.port & AUDIO_AUX2_OUT)
264 gain = AUDIO_MAX_GAIN;
270 switch (track->num_channels) {
272 if (balance == AUDIO_MID_BALANCE) {
275 } else if (balance < AUDIO_MID_BALANCE) {
277 ratio = 1 - (float) (AUDIO_MID_BALANCE - balance) /
278 (float) AUDIO_MID_BALANCE;
279 volumes[1] = (int) ((float) gain * ratio + 0.5);
282 ratio = 1 - (float) (balance - AUDIO_MID_BALANCE) /
283 (float) AUDIO_MID_BALANCE;
284 volumes[0] = (int) ((float) gain * ratio + 0.5);
292 /* Likewise reset MUTE */
293 if ((sunaudiotrack->track_num == GST_SUNAUDIO_TRACK_OUTPUT
294 && audioinfo.output_muted == 1)
295 || (sunaudiotrack->track_num != GST_SUNAUDIO_TRACK_OUTPUT && gain == 0)) {
297 * If MUTE is set, then gain is always 0, so don't bother
298 * resetting our internal value.
300 track->flags |= GST_MIXER_TRACK_MUTE;
302 sunaudiotrack->gain = gain;
303 sunaudiotrack->balance = balance;
304 track->flags &= ~GST_MIXER_TRACK_MUTE;
309 gst_sunaudiomixer_ctrl_set_volume (GstSunAudioMixerCtrl * mixer,
310 GstMixerTrack * track, gint * volumes)
317 struct audio_info audioinfo;
318 GstSunAudioMixerTrack *sunaudiotrack = GST_SUNAUDIO_MIXER_TRACK (track);
320 l_real_gain = volumes[0];
321 r_real_gain = volumes[1];
323 if (l_real_gain == r_real_gain) {
325 balance = AUDIO_MID_BALANCE;
326 } else if (l_real_gain < r_real_gain) {
328 ratio = (float) l_real_gain / (float) r_real_gain;
330 AUDIO_RIGHT_BALANCE - (int) (ratio * (float) AUDIO_MID_BALANCE + 0.5);
333 ratio = (float) r_real_gain / (float) l_real_gain;
335 AUDIO_LEFT_BALANCE + (int) (ratio * (float) AUDIO_MID_BALANCE + 0.5);
338 sunaudiotrack->gain = gain;
339 sunaudiotrack->balance = balance;
341 if (GST_MIXER_TRACK_HAS_FLAG (track, GST_MIXER_TRACK_MUTE)) {
342 if (sunaudiotrack->track_num == GST_SUNAUDIO_TRACK_OUTPUT) {
344 } else if (gain == 0) {
348 * If the volume is set to a non-zero value for LINE_IN
349 * or MONITOR, then unset MUTE.
351 track->flags &= ~GST_MIXER_TRACK_MUTE;
356 AUDIO_INITINFO (&audioinfo);
358 switch (sunaudiotrack->track_num) {
359 case GST_SUNAUDIO_TRACK_OUTPUT:
360 audioinfo.play.gain = gain;
361 audioinfo.play.balance = balance;
363 case GST_SUNAUDIO_TRACK_RECORD:
364 audioinfo.record.gain = gain;
365 audioinfo.record.balance = balance;
367 case GST_SUNAUDIO_TRACK_MONITOR:
368 audioinfo.monitor_gain = gain;
369 audioinfo.record.balance = balance;
375 g_return_if_fail (mixer->mixer_fd != -1);
377 if (ioctl (mixer->mixer_fd, AUDIO_SETINFO, &audioinfo) < 0) {
378 g_warning ("Error setting audio device volume");
384 gst_sunaudiomixer_ctrl_set_mute (GstSunAudioMixerCtrl * mixer,
385 GstMixerTrack * track, gboolean mute)
387 struct audio_info audioinfo;
388 struct audio_info oldinfo;
389 GstSunAudioMixerTrack *sunaudiotrack = GST_SUNAUDIO_MIXER_TRACK (track);
390 gint volume, balance;
392 AUDIO_INITINFO (&audioinfo);
394 if (ioctl (mixer->mixer_fd, AUDIO_GETINFO, &oldinfo) < 0) {
395 g_warning ("Error getting audio device volume");
401 track->flags |= GST_MIXER_TRACK_MUTE;
403 volume = sunaudiotrack->gain;
404 track->flags &= ~GST_MIXER_TRACK_MUTE;
407 balance = sunaudiotrack->balance;
409 switch (sunaudiotrack->track_num) {
410 case GST_SUNAUDIO_TRACK_OUTPUT:
412 audioinfo.output_muted = 1;
414 audioinfo.output_muted = 0;
416 audioinfo.play.gain = volume;
417 audioinfo.play.balance = balance;
419 case GST_SUNAUDIO_TRACK_RECORD:
420 audioinfo.record.gain = volume;
421 audioinfo.record.balance = balance;
423 case GST_SUNAUDIO_TRACK_MONITOR:
424 audioinfo.monitor_gain = volume;
425 audioinfo.record.balance = balance;
427 case GST_SUNAUDIO_TRACK_SPEAKER:
429 audioinfo.play.port = oldinfo.play.port & ~AUDIO_SPEAKER;
431 audioinfo.play.port = oldinfo.play.port | AUDIO_SPEAKER;
434 case GST_SUNAUDIO_TRACK_HP:
436 audioinfo.play.port = oldinfo.play.port & ~AUDIO_HEADPHONE;
438 audioinfo.play.port = oldinfo.play.port | AUDIO_HEADPHONE;
441 case GST_SUNAUDIO_TRACK_LINEOUT:
443 audioinfo.play.port = oldinfo.play.port & ~AUDIO_LINE_OUT;
445 audioinfo.play.port = oldinfo.play.port | AUDIO_LINE_OUT;
448 case GST_SUNAUDIO_TRACK_SPDIFOUT:
450 audioinfo.play.port = oldinfo.play.port & ~AUDIO_SPDIF_OUT;
452 audioinfo.play.port = oldinfo.play.port | AUDIO_SPDIF_OUT;
455 case GST_SUNAUDIO_TRACK_AUX1OUT:
457 audioinfo.play.port = oldinfo.play.port & ~AUDIO_AUX1_OUT;
459 audioinfo.play.port = oldinfo.play.port | AUDIO_AUX1_OUT;
462 case GST_SUNAUDIO_TRACK_AUX2OUT:
464 audioinfo.play.port = oldinfo.play.port & ~AUDIO_AUX2_OUT;
466 audioinfo.play.port = oldinfo.play.port | AUDIO_AUX2_OUT;
473 if (audioinfo.play.port != ((unsigned) ~0)) {
474 /* mask off ports we can't modify. Hack for broken drivers where mod_ports == 0 */
475 if (oldinfo.play.mod_ports != 0) {
476 audioinfo.play.port &= oldinfo.play.mod_ports;
477 /* and add in any that are forced to be on */
478 audioinfo.play.port |= (oldinfo.play.port & ~oldinfo.play.mod_ports);
481 g_return_if_fail (mixer->mixer_fd != -1);
483 if (audioinfo.play.port != (guint) (-1) &&
484 audioinfo.play.port != oldinfo.play.port)
485 GST_LOG_OBJECT (mixer, "Changing play port mask to 0x%08x",
486 audioinfo.play.port);
488 if (ioctl (mixer->mixer_fd, AUDIO_SETINFO, &audioinfo) < 0) {
489 g_warning ("Error setting audio settings");
495 gst_sunaudiomixer_ctrl_set_record (GstSunAudioMixerCtrl * mixer,
496 GstMixerTrack * track, gboolean record)
501 gst_sunaudiomixer_ctrl_set_option (GstSunAudioMixerCtrl * mixer,
502 GstMixerOptions * options, gchar * value)
504 struct audio_info audioinfo;
505 GstMixerTrack *track;
506 GstSunAudioMixerOptions *opts;
510 g_return_if_fail (mixer != NULL);
511 g_return_if_fail (mixer->mixer_fd != -1);
512 g_return_if_fail (value != NULL);
513 g_return_if_fail (GST_IS_SUNAUDIO_MIXER_OPTIONS (options));
515 track = GST_MIXER_TRACK (options);
516 opts = GST_SUNAUDIO_MIXER_OPTIONS (options);
518 if (opts->track_num != GST_SUNAUDIO_TRACK_RECSRC) {
519 g_warning ("set_option not supported on track %s", track->label);
523 q = g_quark_try_string (value);
525 g_warning ("unknown option '%s'", value);
529 for (i = 0; i < 8; i++) {
530 if (opts->names[i] == q) {
535 if (((1 << (i)) & opts->avail) == 0) {
536 g_warning ("Record port %s not available", g_quark_to_string (q));
540 AUDIO_INITINFO (&audioinfo);
541 audioinfo.record.port = (1 << (i));
543 if (ioctl (mixer->mixer_fd, AUDIO_SETINFO, &audioinfo) < 0) {
544 g_warning ("Error setting audio record port");
549 gst_sunaudiomixer_ctrl_get_option (GstSunAudioMixerCtrl * mixer,
550 GstMixerOptions * options)
552 GstMixerTrack *track;
553 GstSunAudioMixerOptions *opts;
554 struct audio_info audioinfo;
557 g_return_val_if_fail (mixer != NULL, NULL);
558 g_return_val_if_fail (mixer->fd != -1, NULL);
559 g_return_val_if_fail (GST_IS_SUNAUDIO_MIXER_OPTIONS (options), NULL);
561 track = GST_MIXER_TRACK (options);
562 opts = GST_SUNAUDIO_MIXER_OPTIONS (options);
564 g_return_val_if_fail (opts->track_num == GST_SUNAUDIO_TRACK_RECSRC, NULL);
566 if (ioctl (mixer->mixer_fd, AUDIO_GETINFO, &audioinfo) < 0) {
567 g_warning ("Error getting audio device settings");
571 for (i = 0; i < 8; i++) {
572 if ((1 << i) == audioinfo.record.port) {
573 const gchar *s = g_quark_to_string (opts->names[i]);
574 GST_DEBUG_OBJECT (mixer, "Getting value for option %d: %s",
580 GST_DEBUG_OBJECT (mixer, "Unable to get value for option %d",
583 g_warning ("Record port value %d seems illegal", audioinfo.record.port);