Commit d2e8f641 authored by Takashi Iwai's avatar Takashi Iwai

ALSA: usb-audio: Explicitly set up the clock selector

In the current code, we have some assumption that the audio clock
selector has been set up implicitly and don't want to touch it unless
it's really needed for the fallback autoclock setup.  This works for
most devices but some seem having a problem.  Partially this was
covered for the devices with a single connector at the initialization
phase (commit 086b957c "ALSA: usb-audio: Skip the clock selector
inquiry for single connections"), but also there are cases where the
wrong clock set up is kept silently.  The latter seems to be the cause
of the noises on Behringer devices.

In this patch, we explicitly set up the audio clock selector whenever
the appropriate node is found.
Reported-by: default avatarGeraldo Nascimento <geraldogabriel@gmail.com>
BugLink: https://bugzilla.kernel.org/show_bug.cgi?id=199327
Link: https://lore.kernel.org/r/CAEsQvcvF7LnO8PxyyCxuRCx=7jNeSCvFAd-+dE0g_rd1rOxxdw@mail.gmail.com
Cc: <stable@vger.kernel.org>
Link: https://lore.kernel.org/r/20210413084152.32325-1-tiwai@suse.deSigned-off-by: default avatarTakashi Iwai <tiwai@suse.de>
parent d91cbe83
...@@ -296,7 +296,7 @@ static int __uac_clock_find_source(struct snd_usb_audio *chip, ...@@ -296,7 +296,7 @@ static int __uac_clock_find_source(struct snd_usb_audio *chip,
selector = snd_usb_find_clock_selector(chip->ctrl_intf, entity_id); selector = snd_usb_find_clock_selector(chip->ctrl_intf, entity_id);
if (selector) { if (selector) {
int ret, i, cur; int ret, i, cur, err;
if (selector->bNrInPins == 1) { if (selector->bNrInPins == 1) {
ret = 1; ret = 1;
...@@ -324,13 +324,17 @@ static int __uac_clock_find_source(struct snd_usb_audio *chip, ...@@ -324,13 +324,17 @@ static int __uac_clock_find_source(struct snd_usb_audio *chip,
ret = __uac_clock_find_source(chip, fmt, ret = __uac_clock_find_source(chip, fmt,
selector->baCSourceID[ret - 1], selector->baCSourceID[ret - 1],
visited, validate); visited, validate);
if (ret > 0) {
err = uac_clock_selector_set_val(chip, entity_id, cur);
if (err < 0)
return err;
}
if (!validate || ret > 0 || !chip->autoclock) if (!validate || ret > 0 || !chip->autoclock)
return ret; return ret;
/* The current clock source is invalid, try others. */ /* The current clock source is invalid, try others. */
for (i = 1; i <= selector->bNrInPins; i++) { for (i = 1; i <= selector->bNrInPins; i++) {
int err;
if (i == cur) if (i == cur)
continue; continue;
...@@ -396,7 +400,7 @@ static int __uac3_clock_find_source(struct snd_usb_audio *chip, ...@@ -396,7 +400,7 @@ static int __uac3_clock_find_source(struct snd_usb_audio *chip,
selector = snd_usb_find_clock_selector_v3(chip->ctrl_intf, entity_id); selector = snd_usb_find_clock_selector_v3(chip->ctrl_intf, entity_id);
if (selector) { if (selector) {
int ret, i, cur; int ret, i, cur, err;
/* the entity ID we are looking for is a selector. /* the entity ID we are looking for is a selector.
* find out what it currently selects */ * find out what it currently selects */
...@@ -418,6 +422,12 @@ static int __uac3_clock_find_source(struct snd_usb_audio *chip, ...@@ -418,6 +422,12 @@ static int __uac3_clock_find_source(struct snd_usb_audio *chip,
ret = __uac3_clock_find_source(chip, fmt, ret = __uac3_clock_find_source(chip, fmt,
selector->baCSourceID[ret - 1], selector->baCSourceID[ret - 1],
visited, validate); visited, validate);
if (ret > 0) {
err = uac_clock_selector_set_val(chip, entity_id, cur);
if (err < 0)
return err;
}
if (!validate || ret > 0 || !chip->autoclock) if (!validate || ret > 0 || !chip->autoclock)
return ret; return ret;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment