Commit cbc200bc authored by Clemens Ladisch's avatar Clemens Ladisch Committed by Takashi Iwai

ALSA: usb-audio: disable autopm for MIDI devices

Commit 88a8516a (ALSA: usbaudio: implement USB autosuspend)
introduced autopm for all USB audio/MIDI devices.  However, many MIDI
devices, such as synthesizers, do not merely transmit MIDI messages but
use their MIDI inputs to control other functions.  With autopm, these
devices would get powered down as soon as the last MIDI port device is
closed on the host.

Even some plain MIDI interfaces could get broken: they automatically
send Active Sensing messages while powered up, but as soon as these
messages cease, the receiving device would interpret this as an
accidental disconnection.

Commit f5f16541 (ALSA: usb-audio: Fix missing autopm for MIDI input)
introduced another regression: some devices (e.g. the Roland GAIA SH-01)
are self-powered but do a reset whenever the USB interface's power state
changes.

To work around all this, just disable autopm for all USB MIDI devices.

Reported-by: Laurens Holst
Cc: <stable@vger.kernel.org>
Signed-off-by: default avatarClemens Ladisch <clemens@ladisch.de>
Signed-off-by: default avatarTakashi Iwai <tiwai@suse.de>
parent d240d1dc
...@@ -126,7 +126,6 @@ struct snd_usb_midi { ...@@ -126,7 +126,6 @@ struct snd_usb_midi {
struct snd_usb_midi_in_endpoint *in; struct snd_usb_midi_in_endpoint *in;
} endpoints[MIDI_MAX_ENDPOINTS]; } endpoints[MIDI_MAX_ENDPOINTS];
unsigned long input_triggered; unsigned long input_triggered;
bool autopm_reference;
unsigned int opened[2]; unsigned int opened[2];
unsigned char disconnected; unsigned char disconnected;
unsigned char input_running; unsigned char input_running;
...@@ -1040,7 +1039,6 @@ static int substream_open(struct snd_rawmidi_substream *substream, int dir, ...@@ -1040,7 +1039,6 @@ static int substream_open(struct snd_rawmidi_substream *substream, int dir,
{ {
struct snd_usb_midi* umidi = substream->rmidi->private_data; struct snd_usb_midi* umidi = substream->rmidi->private_data;
struct snd_kcontrol *ctl; struct snd_kcontrol *ctl;
int err;
down_read(&umidi->disc_rwsem); down_read(&umidi->disc_rwsem);
if (umidi->disconnected) { if (umidi->disconnected) {
...@@ -1051,13 +1049,6 @@ static int substream_open(struct snd_rawmidi_substream *substream, int dir, ...@@ -1051,13 +1049,6 @@ static int substream_open(struct snd_rawmidi_substream *substream, int dir,
mutex_lock(&umidi->mutex); mutex_lock(&umidi->mutex);
if (open) { if (open) {
if (!umidi->opened[0] && !umidi->opened[1]) { if (!umidi->opened[0] && !umidi->opened[1]) {
err = usb_autopm_get_interface(umidi->iface);
umidi->autopm_reference = err >= 0;
if (err < 0 && err != -EACCES) {
mutex_unlock(&umidi->mutex);
up_read(&umidi->disc_rwsem);
return -EIO;
}
if (umidi->roland_load_ctl) { if (umidi->roland_load_ctl) {
ctl = umidi->roland_load_ctl; ctl = umidi->roland_load_ctl;
ctl->vd[0].access |= SNDRV_CTL_ELEM_ACCESS_INACTIVE; ctl->vd[0].access |= SNDRV_CTL_ELEM_ACCESS_INACTIVE;
...@@ -1080,8 +1071,6 @@ static int substream_open(struct snd_rawmidi_substream *substream, int dir, ...@@ -1080,8 +1071,6 @@ static int substream_open(struct snd_rawmidi_substream *substream, int dir,
snd_ctl_notify(umidi->card, snd_ctl_notify(umidi->card,
SNDRV_CTL_EVENT_MASK_INFO, &ctl->id); SNDRV_CTL_EVENT_MASK_INFO, &ctl->id);
} }
if (umidi->autopm_reference)
usb_autopm_put_interface(umidi->iface);
} }
} }
mutex_unlock(&umidi->mutex); mutex_unlock(&umidi->mutex);
...@@ -2258,6 +2247,8 @@ int snd_usbmidi_create(struct snd_card *card, ...@@ -2258,6 +2247,8 @@ int snd_usbmidi_create(struct snd_card *card,
return err; return err;
} }
usb_autopm_get_interface_no_resume(umidi->iface);
list_add_tail(&umidi->list, midi_list); list_add_tail(&umidi->list, midi_list);
return 0; return 0;
} }
......
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