Commit 5d15f1eb authored by Takashi Iwai's avatar Takashi Iwai

ALSA: usb-audio: Choose audioformat of a counter-part substream

The implicit feedback mode needs to handle two endpoints and the
choice of the audioformat object for the sync EP is important since
this determines the compatibility of the hw_params.  The current code
uses the same audioformat object if both the main EP and the sync EP
point to the same iface/altsetting.  This was done in consideration of
the non-implicit-fb sync EP handling, and it doesn't match well with
the cases where actually to endpoints are defined in the sameiface /
altsetting like a few Pioneer devices.

Modify snd_usb_find_implicit_fb_sync_format() to pick up the
audioformat that is assigned in the counter-part substreams primarily,
so that the actual capture stream can be opened properly.  We keep the
same audioformat object only as a fallback in case nothing found,
though.

Fixes: 9fddc15e ("ALSA: usb-audio: Factor out the implicit feedback quirk code")
Link: https://lore.kernel.org/r/20210108075219.21463-3-tiwai@suse.deSigned-off-by: default avatarTakashi Iwai <tiwai@suse.de>
parent b2345a8a
...@@ -378,20 +378,19 @@ snd_usb_find_implicit_fb_sync_format(struct snd_usb_audio *chip, ...@@ -378,20 +378,19 @@ snd_usb_find_implicit_fb_sync_format(struct snd_usb_audio *chip,
int stream) int stream)
{ {
struct snd_usb_substream *subs; struct snd_usb_substream *subs;
const struct audioformat *fp, *sync_fmt; const struct audioformat *fp, *sync_fmt = NULL;
int score, high_score; int score, high_score;
/* When sharing the same altset, use the original audioformat */ /* Use the original audioformat as fallback for the shared altset */
if (target->iface == target->sync_iface && if (target->iface == target->sync_iface &&
target->altsetting == target->sync_altsetting) target->altsetting == target->sync_altsetting)
return target; sync_fmt = target;
subs = find_matching_substream(chip, stream, target->sync_ep, subs = find_matching_substream(chip, stream, target->sync_ep,
target->fmt_type); target->fmt_type);
if (!subs) if (!subs)
return NULL; return sync_fmt;
sync_fmt = NULL;
high_score = 0; high_score = 0;
list_for_each_entry(fp, &subs->fmt_list, list) { list_for_each_entry(fp, &subs->fmt_list, list) {
score = match_endpoint_audioformats(subs, fp, score = match_endpoint_audioformats(subs, fp,
......
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