Recoded alsamixer utility to use the latest alsa-lib's API.

This commit is contained in:
Jaroslav Kysela 2001-02-14 12:22:47 +00:00
parent b62252dd77
commit c4212d7496
2 changed files with 191 additions and 180 deletions

View file

@ -3,3 +3,6 @@ LDADD = $(CURSESLIB)
bin_PROGRAMS = alsamixer bin_PROGRAMS = alsamixer
man_MANS = alsamixer.1 man_MANS = alsamixer.1
alsamixer_INCLUDES = -I$(top_srcdir)/include -DCURSESINC="$(CURSESINC)" alsamixer_INCLUDES = -I$(top_srcdir)/include -DCURSESINC="$(CURSESINC)"
#LDFLAGS = -static
#CFLAGS += -g -Wall

View file

@ -1,7 +1,5 @@
#if 0 /* AlsaMixer - Commandline mixer for the ALSA project Copyright (C) 1998,
* 1999 Tim Janik <timj@gtk.org> and Jaroslav Kysela <perex@suse.cz>
/* AlsaMixer - Commandline mixer for the ALSA project
* Copyright (C) 1998, 1999 Tim Janik <timj@gtk.org> and Jaroslav Kysela <perex@suse.cz>
* *
* This program is free software; you can redistribute it and/or * This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License * modify it under the terms of the GNU General Public License
@ -20,6 +18,10 @@
* *
* ChangeLog: * ChangeLog:
* *
* Wed Feb 14 13:08:17 CET 2001 Jaroslav Kysela <perex@suse.cz>
*
* * ported to the latest mixer 0.6.x API (function based)
*
* Fri Jun 23 14:10:00 MEST 2000 Jaroslav Kysela <perex@suse.cz> * Fri Jun 23 14:10:00 MEST 2000 Jaroslav Kysela <perex@suse.cz>
* *
* * ported to new mixer 0.6.x API (simple control) * * ported to new mixer 0.6.x API (simple control)
@ -200,14 +202,6 @@ enum {
MIXER_ELEM_END MIXER_ELEM_END
}; };
/* channel mask for each type */
static int mixer_elem_mask[] = {
(SND_MIXER_SCHN_MASK_FRONT_LEFT | SND_MIXER_SCHN_MASK_FRONT_RIGHT),
(SND_MIXER_SCHN_MASK_REAR_LEFT | SND_MIXER_SCHN_MASK_REAR_RIGHT),
SND_MIXER_SCHN_MASK_FRONT_CENTER,
SND_MIXER_SCHN_MASK_WOOFER,
};
/* left and right channels for each type */ /* left and right channels for each type */
static snd_mixer_selem_channel_id_t mixer_elem_chn[][2] = { static snd_mixer_selem_channel_id_t mixer_elem_chn[][2] = {
{ SND_MIXER_SCHN_FRONT_LEFT, SND_MIXER_SCHN_FRONT_RIGHT }, { SND_MIXER_SCHN_FRONT_LEFT, SND_MIXER_SCHN_FRONT_RIGHT },
@ -216,8 +210,8 @@ static snd_mixer_selem_channel_id_t mixer_elem_chn[][2] = {
{ SND_MIXER_SCHN_WOOFER, SND_MIXER_SCHN_UNKNOWN }, { SND_MIXER_SCHN_WOOFER, SND_MIXER_SCHN_UNKNOWN },
}; };
static snd_mixer_selem_id_t *mixer_sid = NULL; static void *mixer_sid = NULL;
static int mixer_n_scontrols = 0; static int mixer_n_selems = 0;
/* split scontrols */ /* split scontrols */
static int mixer_n_elems = 0; static int mixer_n_elems = 0;
@ -471,11 +465,14 @@ mixer_conv(int val, int omin, int omax, int nmin, int nmax)
} }
static int static int
mixer_calc_volume(snd_mixer_selem_t *scontrol, int vol, snd_mixer_selem_channel_id_t chn) mixer_calc_volume(snd_mixer_selem_info_t *info,
snd_mixer_selem_value_t *value,
int vol,
snd_mixer_selem_channel_id_t chn)
{ {
int vol1; int vol1;
long min = snd_mixer_selem_get_min(scontrol); long min = snd_mixer_selem_info_get_min(info);
long max = snd_mixer_selem_get_max(scontrol); long max = snd_mixer_selem_info_get_max(info);
vol1 = (vol < 0) ? -vol : vol; vol1 = (vol < 0) ? -vol : vol;
if (vol1 > 0) { if (vol1 > 0) {
if (vol1 > 100) if (vol1 > 100)
@ -487,7 +484,7 @@ mixer_calc_volume(snd_mixer_selem_t *scontrol, int vol, snd_mixer_selem_channel_
if (vol < 0) if (vol < 0)
vol1 = -vol1; vol1 = -vol1;
} }
vol1 += snd_mixer_selem_get_channel_volume(scontrol, chn); vol1 += snd_mixer_selem_value_get_volume(value, chn);
return CLAMP(vol1, min, max); return CLAMP(vol1, min, max);
} }
@ -496,60 +493,68 @@ mixer_calc_volume(snd_mixer_selem_t *scontrol, int vol, snd_mixer_selem_channel_
static void static void
mixer_write_cbar (int elem_index) mixer_write_cbar (int elem_index)
{ {
snd_mixer_selem_t *scontrol; snd_mixer_elem_t *elem;
snd_mixer_selem_info_t *info;
snd_mixer_selem_value_t *value;
int vleft, vright, vbalance; int vleft, vright, vbalance;
int type; int type;
snd_mixer_selem_id_t *sid;
snd_mixer_selem_channel_id_t chn_left, chn_right, chn; snd_mixer_selem_channel_id_t chn_left, chn_right, chn;
int err, changed; int err, changed;
snd_mixer_selem_alloca(&scontrol); snd_mixer_selem_info_alloca(&info);
snd_mixer_selem_value_alloca(&value);
if (mixer_sid == NULL) if (mixer_sid == NULL)
return; return;
scontrol.sid = mixer_sid[mixer_grpidx[elem_index]];
if ((err = snd_mixer_selem_read (mixer_handle, &scontrol)) < 0)
CHECK_ABORT (ERR_FCN, "snd_mixer_selem_read()", err);
sid = (snd_mixer_selem_id_t *)(((char *)mixer_sid) + snd_mixer_selem_id_sizeof() * mixer_grpidx[elem_index]);
elem = snd_mixer_find_selem(mixer_handle, sid);
if (elem == NULL)
CHECK_ABORT (ERR_FCN, "snd_mixer_find_selem()", -EINVAL);
if ((err = snd_mixer_selem_info(elem, info)) < 0)
CHECK_ABORT (ERR_FCN, "snd_mixer_selem_info()", err);
if ((err = snd_mixer_selem_read(elem, value)) < 0)
CHECK_ABORT (ERR_FCN, "snd_mixer_selem_read()", err);
type = mixer_type[elem_index]; type = mixer_type[elem_index];
chn_left = mixer_elem_chn[type][MIXER_CHN_LEFT]; chn_left = mixer_elem_chn[type][MIXER_CHN_LEFT];
if (! (scontrol.channels & (1 << snd_enum_to_int(chn_left)))) if (!snd_mixer_selem_info_has_channel(info, chn_left))
return; /* ..??.. */ return; /* ..??.. */
chn_right = mixer_elem_chn[type][MIXER_CHN_RIGHT]; chn_right = mixer_elem_chn[type][MIXER_CHN_RIGHT];
if (chn_right != SND_MIXER_SCHN_UNKNOWN && ! (scontrol.channels & (1 << snd_enum_to_int(chn_right)))) if (chn_right != SND_MIXER_SCHN_UNKNOWN && ! snd_mixer_selem_info_has_channel(info, chn_right))
chn_right = SND_MIXER_SCHN_UNKNOWN; chn_right = SND_MIXER_SCHN_UNKNOWN;
changed = 0; changed = 0;
/* volue /* volume
*/ */
if ((mixer_volume_delta[MIXER_CHN_LEFT] || if ((mixer_volume_delta[MIXER_CHN_LEFT] ||
mixer_volume_delta[MIXER_CHN_RIGHT] || mixer_volume_delta[MIXER_CHN_RIGHT] ||
mixer_balance_volumes) && mixer_balance_volumes) &&
(scontrol.caps & SND_MIXER_SCTCAP_VOLUME)) { snd_mixer_selem_info_has_volume(info)) {
int mono = int mono =
(chn_right == SND_MIXER_SCHN_UNKNOWN || (scontrol.caps & SND_MIXER_SCTCAP_JOINTLY_VOLUME)); (chn_right == SND_MIXER_SCHN_UNKNOWN ||
snd_mixer_selem_info_has_joined_volume(info));
if (mono && !mixer_volume_delta[MIXER_CHN_LEFT]) if (mono && !mixer_volume_delta[MIXER_CHN_LEFT])
mixer_volume_delta[MIXER_CHN_LEFT] = mixer_volume_delta[MIXER_CHN_RIGHT]; mixer_volume_delta[MIXER_CHN_LEFT] = mixer_volume_delta[MIXER_CHN_RIGHT];
vleft = mixer_calc_volume(&scontrol, mixer_volume_delta[MIXER_CHN_LEFT], chn_left); vleft = mixer_calc_volume(info, value, mixer_volume_delta[MIXER_CHN_LEFT], chn_left);
vbalance = vleft; vbalance = vleft;
if (! mono) { if (! mono) {
vright = mixer_calc_volume(&scontrol, mixer_volume_delta[MIXER_CHN_RIGHT], chn_right); vright = mixer_calc_volume(info, value, mixer_volume_delta[MIXER_CHN_RIGHT], chn_right);
vbalance += vright; vbalance += vright;
vbalance /= 2; vbalance /= 2;
} else } else
vright = vleft; vright = vleft;
if (vleft >= 0 && vright >= 0) { if (vleft >= 0 && vright >= 0) {
if (scontrol.caps & SND_MIXER_SCTCAP_JOINTLY_VOLUME) { if (snd_mixer_selem_info_has_joined_volume(info)) {
for (chn = 0; chn < SND_MIXER_SCHN_LAST; snd_enum_incr(chn)) { for (chn = 0; chn < SND_MIXER_SCHN_LAST; snd_enum_incr(chn))
int c = snd_enum_to_int(chn); if (snd_mixer_selem_info_has_channel(info, chn))
if (scontrol.channels & (1 << c)) snd_mixer_selem_value_set_volume(value, chn, vleft);
scontrol.volume.values[c] = vleft;
}
} else { } else {
if (mixer_balance_volumes) if (mixer_balance_volumes)
vleft = vright = vbalance; vleft = vright = vbalance;
scontrol.volume.values[snd_enum_to_int(chn_left)] = vleft; snd_mixer_selem_value_set_volume(value, chn_left, vleft);
if (! mono) if (! mono)
scontrol.volume.values[snd_enum_to_int(chn_right)] = vright; snd_mixer_selem_value_set_volume(value, chn_right, vright);
} }
changed = 1; changed = 1;
} }
@ -559,16 +564,15 @@ mixer_write_cbar (int elem_index)
/* mute /* mute
*/ */
if (mixer_toggle_mute && (scontrol.caps & SND_MIXER_SCTCAP_MUTE)) { if (mixer_toggle_mute && snd_mixer_selem_info_has_mute(info)) {
scontrol.mute &= scontrol.channels; if (snd_mixer_selem_info_has_joined_mute(info))
if (scontrol.caps & SND_MIXER_SCTCAP_JOINTLY_MUTE) snd_mixer_selem_value_set_mute_all(value, snd_mixer_selem_value_get_mute(value, chn_left) ? 0 : 1);
scontrol.mute = scontrol.mute ? 0 : scontrol.channels;
else { else {
if (mixer_toggle_mute & MIXER_MASK_LEFT) if (mixer_toggle_mute & MIXER_MASK_LEFT)
scontrol.mute ^= (1 << snd_enum_to_int(chn_left)); snd_mixer_selem_value_set_mute(value, chn_left, snd_mixer_selem_value_get_mute(value, chn_left) ? 0 : 1);
if (chn_right != SND_MIXER_SCHN_UNKNOWN && if (chn_right != SND_MIXER_SCHN_UNKNOWN &&
(mixer_toggle_mute & MIXER_MASK_RIGHT)) (mixer_toggle_mute & MIXER_MASK_RIGHT))
scontrol.mute ^= (1 << snd_enum_to_int(chn_right)); snd_mixer_selem_value_set_mute(value, chn_right, snd_mixer_selem_value_get_mute(value, chn_right) ? 0 : 1);
} }
changed = 1; changed = 1;
} }
@ -576,22 +580,21 @@ mixer_write_cbar (int elem_index)
/* capture /* capture
*/ */
if (mixer_toggle_capture && (scontrol.caps & SND_MIXER_SCTCAP_CAPTURE)) { if (mixer_toggle_capture && snd_mixer_selem_info_has_capture(info)) {
scontrol.capture &= scontrol.channels; if (snd_mixer_selem_info_has_joined_capture(info))
if (scontrol.caps & SND_MIXER_SCTCAP_JOINTLY_CAPTURE) snd_mixer_selem_value_set_capture_all(value, snd_mixer_selem_value_get_capture(value, chn_left) ? 0 : 1);
scontrol.capture = scontrol.capture ? 0 : scontrol.channels;
else { else {
if (mixer_toggle_capture & MIXER_MASK_LEFT) if (mixer_toggle_capture & MIXER_MASK_LEFT)
scontrol.capture ^= (1 << snd_enum_to_int(chn_left)); snd_mixer_selem_value_set_capture(value, chn_left, snd_mixer_selem_value_get_capture(value, chn_left) ? 0 : 1);
if (chn_right != SND_MIXER_SCHN_UNKNOWN && (mixer_toggle_capture & MIXER_MASK_RIGHT)) if (chn_right != SND_MIXER_SCHN_UNKNOWN && (mixer_toggle_capture & MIXER_MASK_RIGHT))
scontrol.capture ^= (1 << snd_enum_to_int(chn_right)); snd_mixer_selem_value_set_capture(value, chn_right, snd_mixer_selem_value_get_capture(value, chn_right) ? 0 : 1);
} }
changed = 1; changed = 1;
} }
mixer_toggle_capture = 0; mixer_toggle_capture = 0;
if (changed) { if (changed) {
if ((err = snd_mixer_selem_write (mixer_handle, &scontrol)) < 0) if ((err = snd_mixer_selem_write (elem, value)) < 0)
CHECK_ABORT (ERR_FCN, "snd_mixer_selem_write()", err); CHECK_ABORT (ERR_FCN, "snd_mixer_selem_write()", err);
} }
} }
@ -602,43 +605,50 @@ mixer_update_cbar (int elem_index)
{ {
char string[128], string1[64]; char string[128], string1[64];
int err, dc; int err, dc;
snd_mixer_selem_t scontrol; snd_mixer_elem_t *elem;
snd_mixer_selem_info_t *info;
snd_mixer_selem_value_t *value;
int vleft, vright; int vleft, vright;
int type; int type;
snd_mixer_selem_id_t *sid;
snd_mixer_selem_channel_id_t chn_left, chn_right; snd_mixer_selem_channel_id_t chn_left, chn_right;
int x, y, i; int x, y, i;
int c_left, c_right; snd_mixer_selem_info_alloca(&info);
snd_mixer_selem_value_alloca(&value);
/* set new scontrol indices and read info /* set new scontrol indices and read info
*/ */
bzero(&scontrol, sizeof(scontrol));
if (mixer_sid == NULL) if (mixer_sid == NULL)
return; return;
scontrol.sid = mixer_sid[mixer_grpidx[elem_index]];
if ((err = snd_mixer_selem_read (mixer_handle, &scontrol)) < 0) sid = (snd_mixer_selem_id_t *)(((char *)mixer_sid) + snd_mixer_selem_id_sizeof() * mixer_grpidx[elem_index]);
elem = snd_mixer_find_selem(mixer_handle, sid);
if (elem == NULL)
CHECK_ABORT (ERR_FCN, "snd_mixer_find_selem()", -EINVAL);
if ((err = snd_mixer_selem_info(elem, info)) < 0)
CHECK_ABORT (ERR_FCN, "snd_mixer_selem_info()", err);
if ((err = snd_mixer_selem_read(elem, value)) < 0)
CHECK_ABORT (ERR_FCN, "snd_mixer_selem_read()", err); CHECK_ABORT (ERR_FCN, "snd_mixer_selem_read()", err);
type = mixer_type[elem_index]; type = mixer_type[elem_index];
chn_left = mixer_elem_chn[type][MIXER_CHN_LEFT]; chn_left = mixer_elem_chn[type][MIXER_CHN_LEFT];
if (! (scontrol.channels & (1 << snd_enum_to_int(chn_left)))) if (!snd_mixer_selem_info_has_channel(info, chn_left))
return; /* ..??.. */ return; /* ..??.. */
chn_right = mixer_elem_chn[type][MIXER_CHN_RIGHT]; chn_right = mixer_elem_chn[type][MIXER_CHN_RIGHT];
if (chn_right != SND_MIXER_SCHN_UNKNOWN && if (chn_right != SND_MIXER_SCHN_UNKNOWN && ! snd_mixer_selem_info_has_channel(info, chn_right))
! (scontrol.channels & (1 << snd_enum_to_int(chn_right))))
chn_right = SND_MIXER_SCHN_UNKNOWN; chn_right = SND_MIXER_SCHN_UNKNOWN;
c_left = snd_enum_to_int(chn_left);
c_right = snd_enum_to_int(chn_right);
/* first, read values for the numbers to be displayed vleft = snd_mixer_selem_value_get_volume(value, chn_left);
*/ vleft = mixer_conv(vleft,
if ((err = snd_mixer_selem_read (mixer_handle, &scontrol)) < 0) snd_mixer_selem_info_get_min(info),
CHECK_ABORT (ERR_FCN, "snd_mixer_selem_read()", err); snd_mixer_selem_info_get_max(info),
0, 100);
vleft = scontrol.volume.values[c_left];
vleft = mixer_conv(vleft, scontrol.min, scontrol.max, 0, 100);
if (chn_right != SND_MIXER_SCHN_UNKNOWN) { if (chn_right != SND_MIXER_SCHN_UNKNOWN) {
vright = scontrol.volume.values[c_right]; vright = snd_mixer_selem_value_get_volume(value, chn_right);
vright = mixer_conv(vright, scontrol.min, scontrol.max, 0, 100); vright = mixer_conv(vright,
snd_mixer_selem_info_get_min(info),
snd_mixer_selem_info_get_max(info),
0, 100);
} else { } else {
vright = vleft; vright = vleft;
} }
@ -653,9 +663,9 @@ mixer_update_cbar (int elem_index)
for (i = 0; i < 63; i++) for (i = 0; i < 63; i++)
string1[i] = ' '; string1[i] = ' ';
string1[63] = '\0'; string1[63] = '\0';
strcpy(string, scontrol.sid.name); strcpy(string, snd_mixer_selem_id_get_name(sid));
if (scontrol.sid.index > 0) if (snd_mixer_selem_id_get_index(sid) > 0)
sprintf(string + strlen(string), " %i", scontrol.sid.index); sprintf(string + strlen(string), " %i", snd_mixer_selem_id_get_index(sid));
string[63] = '\0'; string[63] = '\0';
strncpy(string1, string, strlen(string)); strncpy(string1, string, strlen(string));
addstr(string1); addstr(string1);
@ -669,10 +679,10 @@ mixer_update_cbar (int elem_index)
/* channel bar name /* channel bar name
*/ */
mixer_dc (elem_index == mixer_focus_elem ? DC_CBAR_FOCUS_LABEL : DC_CBAR_LABEL); mixer_dc (elem_index == mixer_focus_elem ? DC_CBAR_FOCUS_LABEL : DC_CBAR_LABEL);
if (scontrol.sid.index > 0) if (snd_mixer_selem_id_get_index(sid) > 0)
sprintf(string1, "%s %d", scontrol.sid.name, scontrol.sid.index); sprintf(string1, "%s %d", snd_mixer_selem_id_get_name(sid), snd_mixer_selem_id_get_index(sid));
else else
strcpy(string1, scontrol.sid.name); strcpy(string1, snd_mixer_selem_id_get_name(sid));
string1[8] = 0; string1[8] = 0;
for (i = 0; i < 8; i++) for (i = 0; i < 8; i++)
{ {
@ -733,10 +743,10 @@ mixer_update_cbar (int elem_index)
mvaddstr (y, x, " "); mvaddstr (y, x, " ");
mixer_dc (DC_CBAR_FRAME); mixer_dc (DC_CBAR_FRAME);
mvaddch (y, x + 2, ACS_ULCORNER); mvaddch (y, x + 2, ACS_ULCORNER);
dc = scontrol.mute & (1 << c_left) ? DC_CBAR_MUTE : DC_CBAR_NOMUTE; dc = snd_mixer_selem_value_get_mute(value, chn_left) ? DC_CBAR_MUTE : DC_CBAR_NOMUTE;
mvaddch (y, x + 3, mixer_dc (dc)); mvaddch (y, x + 3, mixer_dc (dc));
if (chn_right != SND_MIXER_SCHN_UNKNOWN) if (chn_right != SND_MIXER_SCHN_UNKNOWN)
dc = scontrol.mute & (1 << c_right) ? DC_CBAR_MUTE : DC_CBAR_NOMUTE; dc = snd_mixer_selem_value_get_mute(value, chn_right) ? DC_CBAR_MUTE : DC_CBAR_NOMUTE;
mvaddch (y, x + 4, mixer_dc (dc)); mvaddch (y, x + 4, mixer_dc (dc));
mixer_dc (DC_CBAR_FRAME); mixer_dc (DC_CBAR_FRAME);
mvaddch (y, x + 5, ACS_URCORNER); mvaddch (y, x + 5, ACS_URCORNER);
@ -744,21 +754,21 @@ mixer_update_cbar (int elem_index)
/* capture input? /* capture input?
*/ */
if ((scontrol.capture & (1 << c_left)) || if ((snd_mixer_selem_value_get_capture(value, chn_left)) ||
(chn_right != SND_MIXER_SCHN_UNKNOWN && (scontrol.capture & (1 << c_right)))) (chn_right != SND_MIXER_SCHN_UNKNOWN && snd_mixer_selem_value_get_capture(value, chn_right)))
{ {
mixer_dc (DC_CBAR_CAPTURE); mixer_dc (DC_CBAR_CAPTURE);
mvaddstr (y, x + 1, "CAPTUR"); mvaddstr (y, x + 1, "CAPTUR");
if (scontrol.capture & (1 << c_left)) { if (snd_mixer_selem_value_get_capture(value, chn_left)) {
mvaddstr (y + 1, x + 1, "L"); mvaddstr (y + 1, x + 1, "L");
if (chn_right == SND_MIXER_SCHN_UNKNOWN) if (chn_right == SND_MIXER_SCHN_UNKNOWN)
mvaddstr (y + 1, x + 6, "R"); mvaddstr (y + 1, x + 6, "R");
} }
if (chn_right != SND_MIXER_SCHN_UNKNOWN && if (chn_right != SND_MIXER_SCHN_UNKNOWN &&
(scontrol.capture & (1 << c_right))) snd_mixer_selem_value_get_capture(value, chn_right))
mvaddstr (y + 1, x + 6, "R"); mvaddstr (y + 1, x + 6, "R");
} }
else if (scontrol.caps & SND_MIXER_SCTCAP_CAPTURE) else if (snd_mixer_selem_info_has_capture(info))
for (i = 0; i < 6; i++) for (i = 0; i < 6; i++)
mvaddch (y, x + 1 + i, mixer_dc (DC_CBAR_NOCAPTURE)); mvaddch (y, x + 1 + i, mixer_dc (DC_CBAR_NOCAPTURE));
else else
@ -1229,8 +1239,14 @@ mixer_init (void)
snd_ctl_close (ctl_handle); snd_ctl_close (ctl_handle);
/* open mixer device /* open mixer device
*/ */
if ((err = snd_mixer_open (&mixer_handle, card_id)) < 0) if ((err = snd_mixer_open (&mixer_handle)) < 0)
mixer_abort (ERR_OPEN, "snd_mixer_open", err); mixer_abort (ERR_FCN, "snd_mixer_open", err);
if ((err = snd_mixer_attach (mixer_handle, card_id)) < 0)
mixer_abort (ERR_FCN, "snd_mixer_attach", err);
if ((err = snd_mixer_selem_register (mixer_handle, NULL)) < 0)
mixer_abort (ERR_FCN, "snd_mixer_selem_register", err);
if ((err = snd_mixer_load (mixer_handle)) < 0)
mixer_abort (ERR_FCN, "snd_mixer_load", err);
/* setup global variables /* setup global variables
*/ */
@ -1241,52 +1257,47 @@ mixer_init (void)
static void static void
mixer_reinit (void) mixer_reinit (void)
{ {
snd_mixer_selem_list_t scontrols; snd_mixer_elem_t *elem;
int idx, err, elem_index, i; int idx, err, elem_index, i, j;
snd_mixer_selem_id_t focus_gid; snd_mixer_selem_id_t *sid;
snd_mixer_selem_id_t *focus_gid;
int focus_type = -1; int focus_type = -1;
snd_mixer_selem_id_alloca(&focus_gid);
if (mixer_sid) { if (mixer_sid) {
focus_gid = mixer_sid[mixer_grpidx[mixer_focus_elem]]; snd_mixer_selem_id_copy(focus_gid, (snd_mixer_selem_id_t *)(((char *)mixer_sid) + snd_mixer_selem_id_sizeof() * mixer_grpidx[mixer_focus_elem]));
focus_type = mixer_type[mixer_focus_elem]; focus_type = mixer_type[mixer_focus_elem];
} }
while (1) { if (mixer_sid != NULL)
bzero(&scontrols, sizeof(scontrols)); free(mixer_sid);
if ((err = snd_mixer_selem_list(mixer_handle, &scontrols)) < 0) mixer_sid = malloc(snd_mixer_selem_id_sizeof() * snd_mixer_elem_count(mixer_handle));
mixer_abort (ERR_FCN, "snd_mixer_selem_list", err); if (mixer_sid == NULL)
mixer_n_scontrols = scontrols.controls; mixer_abort (ERR_FCN, "malloc", 0);
if (mixer_n_scontrols > 0) {
scontrols.controls_request = mixer_n_scontrols; mixer_n_selems = 0;
scontrols.pids = (snd_mixer_selem_id_t *)malloc(sizeof(snd_mixer_selem_id_t) * mixer_n_scontrols); for (elem = snd_mixer_first_elem(mixer_handle); elem; elem = snd_mixer_elem_next(elem)) {
if (scontrols.pids == NULL) sid = (snd_mixer_selem_id_t *)(((char *)mixer_sid) + snd_mixer_selem_id_sizeof() * mixer_n_selems);
mixer_abort (ERR_FCN, "malloc", 0); snd_mixer_selem_get_id(elem, sid);
scontrols.controls_offset = 0; mixer_n_selems++;
scontrols.controls_count = 0;
if ((err = snd_mixer_selem_list(mixer_handle, &scontrols)) < 0)
mixer_abort (ERR_FCN, "snd_mixer_selem_list", err);
if (scontrols.controls > scontrols.controls_count) {
free(scontrols.pids);
continue;
}
}
if (mixer_sid)
free(mixer_sid);
mixer_sid = scontrols.pids;
break;
} }
#if 0
snd_mixer_sort_gid_table(mixer_sid, mixer_n_scontrols, snd_mixer_default_weights);
#endif
mixer_n_elems = 0; mixer_n_elems = 0;
for (idx = 0; idx < mixer_n_scontrols; idx++) { for (idx = 0; idx < mixer_n_selems; idx++) {
snd_mixer_selem_t scontrol; snd_mixer_selem_info_t *info;
bzero(&scontrol, sizeof(scontrol)); snd_mixer_selem_info_alloca(&info);
scontrol.sid = mixer_sid[idx]; sid = (snd_mixer_selem_id_t *)(((char *)mixer_sid) + snd_mixer_selem_id_sizeof() * idx);
if ((err = snd_mixer_selem_read(mixer_handle, &scontrol)) < 0) elem = snd_mixer_find_selem(mixer_handle, sid);
CHECK_ABORT (ERR_FCN, "snd_mixer_selem_read()", 0); if (elem == NULL)
CHECK_ABORT (ERR_FCN, "snd_mixer_find_selem()", -EINVAL);
if ((err = snd_mixer_selem_info(elem, info)) < 0)
CHECK_ABORT (ERR_FCN, "snd_mixer_selem_info()", 0);
for (i = 0; i < MIXER_ELEM_END; i++) { for (i = 0; i < MIXER_ELEM_END; i++) {
if (scontrol.channels & mixer_elem_mask[i]) int ok;
for (j = ok = 0; j < 2; j++) {
if (snd_mixer_selem_info_has_channel(info, mixer_elem_chn[i][j]))
ok++;
}
if (ok)
mixer_n_elems++; mixer_n_elems++;
} }
} }
@ -1302,19 +1313,27 @@ mixer_reinit (void)
if (mixer_grpidx == NULL) if (mixer_grpidx == NULL)
mixer_abort(ERR_FCN, "malloc", 0); mixer_abort(ERR_FCN, "malloc", 0);
elem_index = 0; elem_index = 0;
for (idx = 0; idx < mixer_n_scontrols; idx++) { for (idx = 0; idx < mixer_n_selems; idx++) {
snd_mixer_selem_t scontrol; snd_mixer_selem_info_t *info;
bzero(&scontrol, sizeof(scontrol)); snd_mixer_selem_info_alloca(&info);
scontrol.sid = mixer_sid[idx]; sid = (snd_mixer_selem_id_t *)(((char *)mixer_sid) + snd_mixer_selem_id_sizeof() * idx);
if ((err = snd_mixer_selem_read(mixer_handle, &scontrol)) < 0) elem = snd_mixer_find_selem(mixer_handle, sid);
CHECK_ABORT (ERR_FCN, "snd_mixer_selem_read()", err); if (elem == NULL)
CHECK_ABORT (ERR_FCN, "snd_mixer_find_selem()", -EINVAL);
if ((err = snd_mixer_selem_info(elem, info)) < 0)
CHECK_ABORT (ERR_FCN, "snd_mixer_selem_info()", 0);
for (i = 0; i < MIXER_ELEM_END; i++) { for (i = 0; i < MIXER_ELEM_END; i++) {
if (scontrol.channels & mixer_elem_mask[i]) { int ok;
for (j = ok = 0; j < 2; j++) {
if (snd_mixer_selem_info_has_channel(info, mixer_elem_chn[i][j]))
ok++;
}
if (ok) {
mixer_grpidx[elem_index] = idx; mixer_grpidx[elem_index] = idx;
mixer_type[elem_index] = i; mixer_type[elem_index] = i;
elem_index++; elem_index++;
if (elem_index >= mixer_n_elems) if (elem_index >= mixer_n_elems)
break; break;
} }
} }
} }
@ -1322,7 +1341,11 @@ mixer_reinit (void)
mixer_focus_elem = 0; mixer_focus_elem = 0;
if (focus_type >= 0) { if (focus_type >= 0) {
for (elem_index = 0; elem_index < mixer_n_elems; elem_index++) { for (elem_index = 0; elem_index < mixer_n_elems; elem_index++) {
if (!memcmp(&focus_gid, &mixer_sid[mixer_grpidx[elem_index]], sizeof(focus_gid)) && sid = (snd_mixer_selem_id_t *)(((char *)mixer_sid) + snd_mixer_selem_id_sizeof() * mixer_grpidx[elem_index]);
if (!strcmp(snd_mixer_selem_id_get_name(focus_gid),
snd_mixer_selem_id_get_name(sid)) &&
snd_mixer_selem_id_get_index(focus_gid) ==
snd_mixer_selem_id_get_index(sid) &&
mixer_type[elem_index] == focus_type) { mixer_type[elem_index] == focus_type) {
mixer_focus_elem = elem_index; mixer_focus_elem = elem_index;
break; break;
@ -1410,21 +1433,6 @@ mixer_resize (void)
} }
} }
static void
mixer_callback_rebuild (snd_mixer_t *handle ATTRIBUTE_UNUSED, void *private_data ATTRIBUTE_UNUSED)
{
/* we don't actually need to update the individual channels because
* we redraw the whole screen upon every main iteration anyways.
*/
mixer_reinit ();
}
static void
mixer_callback_scontrol (snd_mixer_t *handle ATTRIBUTE_UNUSED, void *private_data ATTRIBUTE_UNUSED, snd_mixer_selem_id_t *gid ATTRIBUTE_UNUSED)
{
mixer_reinit ();
}
static void static void
mixer_set_delta(int delta) mixer_set_delta(int delta)
{ {
@ -1447,44 +1455,53 @@ static int
mixer_iteration (void) mixer_iteration (void)
{ {
struct timeval delay = { 0, 0 }; struct timeval delay = { 0, 0 };
snd_mixer_simple_callbacks_t callbacks = { 0, }; unsigned int count;
int mixer_fd; struct pollfd *fds;
fd_set rfds; int err;
int finished = 0; int finished = 0;
int key = 0; int key = 0;
int old_view; int old_view;
callbacks.rebuild = mixer_callback_rebuild;
callbacks.value = mixer_callback_scontrol;
callbacks.change = mixer_callback_scontrol;
callbacks.add = mixer_callback_scontrol;
callbacks.remove = mixer_callback_scontrol;
/* setup for select on stdin and the mixer fd */ /* setup for select on stdin and the mixer fd */
mixer_fd = snd_mixer_poll_descriptor (mixer_handle); if ((err = snd_mixer_poll_descriptors_count(mixer_handle, &count)) < 0)
FD_ZERO (&rfds); mixer_abort (ERR_FCN, "snd_mixer_poll_descriptors_count", err);
FD_SET (fileno (stdin), &rfds); fds = calloc(count + 1, sizeof(struct pollfd));
FD_SET (mixer_fd, &rfds); if (fds == NULL)
mixer_abort (ERR_FCN, "malloc", 0);
fds->fd = fileno(stdin);
fds->events = POLLIN;
if ((err = snd_mixer_poll_descriptors(mixer_handle, fds + 1, count)) < 0)
mixer_abort (ERR_FCN, "snd_mixer_poll_descriptors", err);
if (err != count)
mixer_abort (ERR_FCN, "snd_mixer_poll_descriptors (err != count)", 0);
delay.tv_sec = 0; delay.tv_sec = 0;
delay.tv_usec = 0 * 100 * 1000; delay.tv_usec = 0 * 100 * 1000;
finished = select (mixer_fd + 1, &rfds, NULL, NULL, mixer_needs_resize ? &delay : NULL) < 0; finished = poll(fds, count + 1, -1);
/* don't abort on handled signals */ /* don't abort on handled signals */
if (finished && errno == EINTR) if (finished && errno == EINTR)
{ finished = 0;
FD_ZERO (&rfds);
finished = 0;
}
else if (mixer_needs_resize) else if (mixer_needs_resize)
mixer_resize (); mixer_resize ();
if (FD_ISSET (mixer_fd, &rfds)) if (finished) {
snd_mixer_simple_read (mixer_handle, &callbacks); if (fds->revents & POLLIN) {
key = getch ();
finished--;
}
}
if (FD_ISSET (fileno (stdin), &rfds)) if (finished) {
key = getch (); if (fds->revents & POLLIN) {
snd_mixer_handle_events(mixer_handle);
finished = 0;
}
}
finished = 0;
free(fds);
old_view = mixer_view; old_view = mixer_view;
@ -1808,12 +1825,3 @@ main (int argc,
mixer_abort (ERR_NONE, "", 0); mixer_abort (ERR_NONE, "", 0);
}; };
#else
#include <stdlib.h>
#include <stdio.h>
int main(void)
{
printf("Please wait for the new code...\n");
return EXIT_FAILURE;
}
#endif