Skip to content

Commit

Permalink
ALSA: usb-audio: Check connector value on resume
Browse files Browse the repository at this point in the history
BugLink: https://bugs.launchpad.net/bugs/1921757

Rear Mic on Lenovo P620 cannot record after S3, despite that there's no
error and the other two functions of the USB audio, Line In and Line
Out, work just fine.

The mic starts to work again after running userspace app like "alsactl
store". Following the lead, the evidence shows that as soon as connector
status is queried, the mic can work again.

So also check connector value on resume to "wake up" the USB audio to
make it functional.

This can be device specific, however I think this generic approach may
benefit more than one device.

Now the resume callback checks connector, and a new callback,
reset_resume, to also restore switches and volumes.

Suggested-by: Takashi Iwai <tiwai@suse.de>
Signed-off-by: Kai-Heng Feng <kai.heng.feng@canonical.com>
Link: https://lore.kernel.org/r/20210325165918.22593-2-kai.heng.feng@canonical.com
Signed-off-by: Takashi Iwai <tiwai@suse.de>
(cherry picked from commit 44609fc linux-next)
Signed-off-by: Kai-Heng Feng <kai.heng.feng@canonical.com>
Signed-off-by: Timo Aaltonen <timo.aaltonen@canonical.com>
  • Loading branch information
Kai-Heng Feng authored and Timo Aaltonen committed Apr 7, 2021
1 parent bc81b0d commit dd76319
Show file tree
Hide file tree
Showing 3 changed files with 36 additions and 11 deletions.
44 changes: 34 additions & 10 deletions sound/usb/mixer.c
Original file line number Diff line number Diff line change
Expand Up @@ -3671,20 +3671,43 @@ static int restore_mixer_value(struct usb_mixer_elem_list *list)
return 0;
}

static int default_mixer_resume(struct usb_mixer_elem_list *list)
{
struct usb_mixer_elem_info *cval = mixer_elem_list_to_info(list);

/* get connector value to "wake up" the USB audio */
if (cval->val_type == USB_MIXER_BOOLEAN && cval->channels == 1)
get_connector_value(cval, NULL, NULL);

return 0;
}

static int default_mixer_reset_resume(struct usb_mixer_elem_list *list)
{
int err = default_mixer_resume(list);

if (err < 0)
return err;
return restore_mixer_value(list);
}

int snd_usb_mixer_resume(struct usb_mixer_interface *mixer, bool reset_resume)
{
struct usb_mixer_elem_list *list;
usb_mixer_elem_resume_func_t f;
int id, err;

if (reset_resume) {
/* restore cached mixer values */
for (id = 0; id < MAX_ID_ELEMS; id++) {
for_each_mixer_elem(list, mixer, id) {
if (list->resume) {
err = list->resume(list);
if (err < 0)
return err;
}
/* restore cached mixer values */
for (id = 0; id < MAX_ID_ELEMS; id++) {
for_each_mixer_elem(list, mixer, id) {
if (reset_resume)
f = list->reset_resume;
else
f = list->resume;
if (f) {
err = f(list);
if (err < 0)
return err;
}
}
}
Expand All @@ -3703,6 +3726,7 @@ void snd_usb_mixer_elem_init_std(struct usb_mixer_elem_list *list,
list->id = unitid;
list->dump = snd_usb_mixer_dump_cval;
#ifdef CONFIG_PM
list->resume = restore_mixer_value;
list->resume = default_mixer_resume;
list->reset_resume = default_mixer_reset_resume;
#endif
}
1 change: 1 addition & 0 deletions sound/usb/mixer.h
Original file line number Diff line number Diff line change
Expand Up @@ -69,6 +69,7 @@ struct usb_mixer_elem_list {
bool is_std_info;
usb_mixer_elem_dump_func_t dump;
usb_mixer_elem_resume_func_t resume;
usb_mixer_elem_resume_func_t reset_resume;
};

/* iterate over mixer element list of the given unit id */
Expand Down
2 changes: 1 addition & 1 deletion sound/usb/mixer_quirks.c
Original file line number Diff line number Diff line change
Expand Up @@ -151,7 +151,7 @@ static int add_single_ctl_with_resume(struct usb_mixer_interface *mixer,
*listp = list;
list->mixer = mixer;
list->id = id;
list->resume = resume;
list->reset_resume = resume;
kctl = snd_ctl_new1(knew, list);
if (!kctl) {
kfree(list);
Expand Down

0 comments on commit dd76319

Please sign in to comment.