mirror of
https://gitlab.freedesktop.org/pulseaudio/pulseaudio.git
synced 2025-10-31 22:25:33 -04:00
alsa: mixer - more clever alias cache implementation
The hw: device can be addressed using the card index (hw:0) or the card identifier (ASCII string - hw:Loopback). Both mixers are equal. The previous code was fine for the mixers without the UCM private prefixes (_ucmXXXX). Make code more robust, create two aliased mixer structures in the mixers array. Signed-off-by: Jaroslav Kysela <perex@perex.cz> Part-of: <https://gitlab.freedesktop.org/pulseaudio/pulseaudio/-/merge_requests/673>
This commit is contained in:
parent
1883355f1b
commit
f5c8b82c3b
2 changed files with 67 additions and 45 deletions
|
|
@ -103,8 +103,8 @@ struct pa_alsa_setting {
|
||||||
|
|
||||||
/* An entry for one ALSA mixer */
|
/* An entry for one ALSA mixer */
|
||||||
struct pa_alsa_mixer {
|
struct pa_alsa_mixer {
|
||||||
|
struct pa_alsa_mixer *alias;
|
||||||
snd_mixer_t *mixer_handle;
|
snd_mixer_t *mixer_handle;
|
||||||
int card_index;
|
|
||||||
pa_alsa_fdlist *fdl;
|
pa_alsa_fdlist *fdl;
|
||||||
bool used_for_probe_only:1;
|
bool used_for_probe_only:1;
|
||||||
};
|
};
|
||||||
|
|
|
||||||
|
|
@ -1687,14 +1687,14 @@ static int mixer_class_event(snd_mixer_class_t *class, unsigned int mask,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int prepare_mixer(snd_mixer_t *mixer, const char *dev) {
|
static int prepare_mixer(snd_mixer_t *mixer, const char *dev, snd_hctl_t *hctl) {
|
||||||
int err;
|
int err;
|
||||||
snd_mixer_class_t *class;
|
snd_mixer_class_t *class;
|
||||||
|
|
||||||
pa_assert(mixer);
|
pa_assert(mixer);
|
||||||
pa_assert(dev);
|
pa_assert(dev);
|
||||||
|
|
||||||
if ((err = snd_mixer_attach(mixer, dev)) < 0) {
|
if ((err = snd_mixer_attach_hctl(mixer, hctl)) < 0) {
|
||||||
pa_log_info("Unable to attach to mixer %s: %s", dev, pa_alsa_strerror(err));
|
pa_log_info("Unable to attach to mixer %s: %s", dev, pa_alsa_strerror(err));
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
@ -1733,37 +1733,29 @@ snd_mixer_t *pa_alsa_open_mixer(pa_hashmap *mixers, int alsa_card_index, bool pr
|
||||||
return m;
|
return m;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pa_alsa_mixer *pa_alsa_create_mixer(pa_hashmap *mixers, const char *dev, snd_mixer_t *m, bool probe) {
|
||||||
|
pa_alsa_mixer *pm;
|
||||||
|
|
||||||
|
pm = pa_xnew0(pa_alsa_mixer, 1);
|
||||||
|
if (pm == NULL)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
pm->used_for_probe_only = probe;
|
||||||
|
pm->mixer_handle = m;
|
||||||
|
pa_hashmap_put(mixers, pa_xstrdup(dev), pm);
|
||||||
|
return pm;
|
||||||
|
}
|
||||||
|
|
||||||
snd_mixer_t *pa_alsa_open_mixer_by_name(pa_hashmap *mixers, const char *dev, bool probe) {
|
snd_mixer_t *pa_alsa_open_mixer_by_name(pa_hashmap *mixers, const char *dev, bool probe) {
|
||||||
int err;
|
int err;
|
||||||
snd_mixer_t *m;
|
snd_mixer_t *m;
|
||||||
|
snd_hctl_t *hctl;
|
||||||
pa_alsa_mixer *pm;
|
pa_alsa_mixer *pm;
|
||||||
char *dev2;
|
|
||||||
void *state;
|
|
||||||
|
|
||||||
pa_assert(mixers);
|
pa_assert(mixers);
|
||||||
pa_assert(dev);
|
pa_assert(dev);
|
||||||
|
|
||||||
pm = pa_hashmap_get(mixers, dev);
|
pm = pa_hashmap_get(mixers, dev);
|
||||||
|
|
||||||
/* The quick card number/index lookup (hw:#)
|
|
||||||
* We already know the card number/index, thus use the mixer
|
|
||||||
* from the cache at first.
|
|
||||||
*/
|
|
||||||
if (!pm && pa_strneq(dev, "hw:", 3)) {
|
|
||||||
const char *s = dev + 3;
|
|
||||||
int card_index;
|
|
||||||
while (*s && *s >= '0' && *s <= '9') s++;
|
|
||||||
if (*s == '\0' && pa_atoi(dev + 3, &card_index) >= 0) {
|
|
||||||
PA_HASHMAP_FOREACH_KV(dev2, pm, mixers, state) {
|
|
||||||
if (pm->card_index == card_index) {
|
|
||||||
dev = dev2;
|
|
||||||
pm = pa_hashmap_get(mixers, dev);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (pm) {
|
if (pm) {
|
||||||
if (!probe)
|
if (!probe)
|
||||||
pm->used_for_probe_only = false;
|
pm->used_for_probe_only = false;
|
||||||
|
|
@ -1775,27 +1767,55 @@ snd_mixer_t *pa_alsa_open_mixer_by_name(pa_hashmap *mixers, const char *dev, boo
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (prepare_mixer(m, dev) >= 0) {
|
err = snd_hctl_open(&hctl, dev, 0);
|
||||||
pm = pa_xnew0(pa_alsa_mixer, 1);
|
if (err < 0) {
|
||||||
if (pm) {
|
pa_log("Error opening hctl device: %s", pa_alsa_strerror(err));
|
||||||
snd_hctl_t *hctl;
|
goto __close;
|
||||||
pm->card_index = -1;
|
|
||||||
/* determine the ALSA card number (index) and store it to card_index */
|
|
||||||
err = snd_mixer_get_hctl(m, dev, &hctl);
|
|
||||||
if (err >= 0) {
|
|
||||||
snd_ctl_card_info_t *info;
|
|
||||||
snd_ctl_card_info_alloca(&info);
|
|
||||||
err = snd_ctl_card_info(snd_hctl_ctl(hctl), info);
|
|
||||||
if (err >= 0)
|
|
||||||
pm->card_index = snd_ctl_card_info_get_card(info);
|
|
||||||
}
|
|
||||||
pm->used_for_probe_only = probe;
|
|
||||||
pm->mixer_handle = m;
|
|
||||||
pa_hashmap_put(mixers, pa_xstrdup(dev), pm);
|
|
||||||
return m;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (prepare_mixer(m, dev, hctl) >= 0) {
|
||||||
|
/* get the ALSA card number (index) and ID (alias) and create two identical mixers */
|
||||||
|
char *p, *dev2, *dev_idx, *dev_id;
|
||||||
|
snd_ctl_card_info_t *info;
|
||||||
|
snd_ctl_card_info_alloca(&info);
|
||||||
|
err = snd_ctl_card_info(snd_hctl_ctl(hctl), info);
|
||||||
|
if (err < 0)
|
||||||
|
goto __std;
|
||||||
|
dev2 = pa_xstrdup(dev);
|
||||||
|
if (dev2 == NULL)
|
||||||
|
goto __close;
|
||||||
|
p = strchr(dev2, ':');
|
||||||
|
/* sanity check - only hw: devices */
|
||||||
|
if (p == NULL || (p - dev2) < 2 || !pa_strneq(p - 2, "hw:", 3)) {
|
||||||
|
pa_xfree(dev2);
|
||||||
|
goto __std;
|
||||||
|
}
|
||||||
|
*p = '\0';
|
||||||
|
dev_idx = pa_sprintf_malloc("%s:%u", dev2, snd_ctl_card_info_get_card(info));
|
||||||
|
dev_id = pa_sprintf_malloc("%s:%s", dev2, snd_ctl_card_info_get_id(info));
|
||||||
|
pa_log_debug("ALSA alias mixers: %s = %s", dev_idx, dev_id);
|
||||||
|
if (dev_idx && dev_id && (strcmp(dev, dev_idx) == 0 || strcmp(dev, dev_id) == 0)) {
|
||||||
|
pm = pa_alsa_create_mixer(mixers, dev_idx, m, probe);
|
||||||
|
if (pm) {
|
||||||
|
pa_alsa_mixer *pm2;
|
||||||
|
pm2 = pa_alsa_create_mixer(mixers, dev_id, m, probe);
|
||||||
|
if (pm2) {
|
||||||
|
pm->alias = pm2;
|
||||||
|
pm2->alias = pm;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pa_xfree(dev_id);
|
||||||
|
pa_xfree(dev_idx);
|
||||||
|
pa_xfree(dev2);
|
||||||
|
__std:
|
||||||
|
if (pm == NULL)
|
||||||
|
pm = pa_alsa_create_mixer(mixers, dev, m, probe);
|
||||||
|
if (pm)
|
||||||
|
return m;
|
||||||
|
}
|
||||||
|
|
||||||
|
__close:
|
||||||
snd_mixer_close(m);
|
snd_mixer_close(m);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
@ -1836,8 +1856,10 @@ void pa_alsa_mixer_free(pa_alsa_mixer *mixer)
|
||||||
{
|
{
|
||||||
if (mixer->fdl)
|
if (mixer->fdl)
|
||||||
pa_alsa_fdlist_free(mixer->fdl);
|
pa_alsa_fdlist_free(mixer->fdl);
|
||||||
if (mixer->mixer_handle)
|
if (mixer->mixer_handle && mixer->alias == NULL)
|
||||||
snd_mixer_close(mixer->mixer_handle);
|
snd_mixer_close(mixer->mixer_handle);
|
||||||
|
if (mixer->alias)
|
||||||
|
mixer->alias->alias = NULL;
|
||||||
pa_xfree(mixer);
|
pa_xfree(mixer);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue