ALSA: usb-audio: Use rwsem for disconnect protection
Replace mutex with rwsem for codec->shutdown protection so that concurrent accesses are allowed. Also add the protection to snd_usb_autosuspend() and snd_usb_autoresume(), too. Reported-by: Matthieu CASTET <matthieu.castet@parrot.com> Cc: <stable@vger.kernel.org> Signed-off-by: Takashi Iwai <tiwai@suse.de>
This commit is contained in:
@@ -339,7 +339,7 @@ static int snd_usb_audio_create(struct usb_device *dev, int idx,
|
|||||||
}
|
}
|
||||||
|
|
||||||
mutex_init(&chip->mutex);
|
mutex_init(&chip->mutex);
|
||||||
mutex_init(&chip->shutdown_mutex);
|
init_rwsem(&chip->shutdown_rwsem);
|
||||||
chip->index = idx;
|
chip->index = idx;
|
||||||
chip->dev = dev;
|
chip->dev = dev;
|
||||||
chip->card = card;
|
chip->card = card;
|
||||||
@@ -560,7 +560,7 @@ static void snd_usb_audio_disconnect(struct usb_device *dev,
|
|||||||
|
|
||||||
card = chip->card;
|
card = chip->card;
|
||||||
mutex_lock(®ister_mutex);
|
mutex_lock(®ister_mutex);
|
||||||
mutex_lock(&chip->shutdown_mutex);
|
down_write(&chip->shutdown_rwsem);
|
||||||
chip->shutdown = 1;
|
chip->shutdown = 1;
|
||||||
chip->num_interfaces--;
|
chip->num_interfaces--;
|
||||||
if (chip->num_interfaces <= 0) {
|
if (chip->num_interfaces <= 0) {
|
||||||
@@ -582,11 +582,11 @@ static void snd_usb_audio_disconnect(struct usb_device *dev,
|
|||||||
snd_usb_mixer_disconnect(p);
|
snd_usb_mixer_disconnect(p);
|
||||||
}
|
}
|
||||||
usb_chip[chip->index] = NULL;
|
usb_chip[chip->index] = NULL;
|
||||||
mutex_unlock(&chip->shutdown_mutex);
|
up_write(&chip->shutdown_rwsem);
|
||||||
mutex_unlock(®ister_mutex);
|
mutex_unlock(®ister_mutex);
|
||||||
snd_card_free_when_closed(card);
|
snd_card_free_when_closed(card);
|
||||||
} else {
|
} else {
|
||||||
mutex_unlock(&chip->shutdown_mutex);
|
up_write(&chip->shutdown_rwsem);
|
||||||
mutex_unlock(®ister_mutex);
|
mutex_unlock(®ister_mutex);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -618,16 +618,20 @@ int snd_usb_autoresume(struct snd_usb_audio *chip)
|
|||||||
{
|
{
|
||||||
int err = -ENODEV;
|
int err = -ENODEV;
|
||||||
|
|
||||||
|
down_read(&chip->shutdown_rwsem);
|
||||||
if (!chip->shutdown && !chip->probing)
|
if (!chip->shutdown && !chip->probing)
|
||||||
err = usb_autopm_get_interface(chip->pm_intf);
|
err = usb_autopm_get_interface(chip->pm_intf);
|
||||||
|
up_read(&chip->shutdown_rwsem);
|
||||||
|
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
void snd_usb_autosuspend(struct snd_usb_audio *chip)
|
void snd_usb_autosuspend(struct snd_usb_audio *chip)
|
||||||
{
|
{
|
||||||
|
down_read(&chip->shutdown_rwsem);
|
||||||
if (!chip->shutdown && !chip->probing)
|
if (!chip->shutdown && !chip->probing)
|
||||||
usb_autopm_put_interface(chip->pm_intf);
|
usb_autopm_put_interface(chip->pm_intf);
|
||||||
|
up_read(&chip->shutdown_rwsem);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int usb_audio_suspend(struct usb_interface *intf, pm_message_t message)
|
static int usb_audio_suspend(struct usb_interface *intf, pm_message_t message)
|
||||||
|
@@ -292,7 +292,7 @@ static int get_ctl_value_v1(struct usb_mixer_elem_info *cval, int request, int v
|
|||||||
err = snd_usb_autoresume(cval->mixer->chip);
|
err = snd_usb_autoresume(cval->mixer->chip);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
return -EIO;
|
return -EIO;
|
||||||
mutex_lock(&chip->shutdown_mutex);
|
down_read(&chip->shutdown_rwsem);
|
||||||
while (timeout-- > 0) {
|
while (timeout-- > 0) {
|
||||||
if (chip->shutdown)
|
if (chip->shutdown)
|
||||||
break;
|
break;
|
||||||
@@ -310,7 +310,7 @@ static int get_ctl_value_v1(struct usb_mixer_elem_info *cval, int request, int v
|
|||||||
err = -EINVAL;
|
err = -EINVAL;
|
||||||
|
|
||||||
out:
|
out:
|
||||||
mutex_unlock(&chip->shutdown_mutex);
|
up_read(&chip->shutdown_rwsem);
|
||||||
snd_usb_autosuspend(cval->mixer->chip);
|
snd_usb_autosuspend(cval->mixer->chip);
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
@@ -337,7 +337,7 @@ static int get_ctl_value_v2(struct usb_mixer_elem_info *cval, int request, int v
|
|||||||
if (ret)
|
if (ret)
|
||||||
goto error;
|
goto error;
|
||||||
|
|
||||||
mutex_lock(&chip->shutdown_mutex);
|
down_read(&chip->shutdown_rwsem);
|
||||||
if (chip->shutdown)
|
if (chip->shutdown)
|
||||||
ret = -ENODEV;
|
ret = -ENODEV;
|
||||||
else {
|
else {
|
||||||
@@ -346,7 +346,7 @@ static int get_ctl_value_v2(struct usb_mixer_elem_info *cval, int request, int v
|
|||||||
USB_RECIP_INTERFACE | USB_TYPE_CLASS | USB_DIR_IN,
|
USB_RECIP_INTERFACE | USB_TYPE_CLASS | USB_DIR_IN,
|
||||||
validx, idx, buf, size);
|
validx, idx, buf, size);
|
||||||
}
|
}
|
||||||
mutex_unlock(&chip->shutdown_mutex);
|
up_read(&chip->shutdown_rwsem);
|
||||||
snd_usb_autosuspend(chip);
|
snd_usb_autosuspend(chip);
|
||||||
|
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
@@ -453,7 +453,7 @@ int snd_usb_mixer_set_ctl_value(struct usb_mixer_elem_info *cval,
|
|||||||
err = snd_usb_autoresume(chip);
|
err = snd_usb_autoresume(chip);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
return -EIO;
|
return -EIO;
|
||||||
mutex_lock(&chip->shutdown_mutex);
|
down_read(&chip->shutdown_rwsem);
|
||||||
while (timeout-- > 0) {
|
while (timeout-- > 0) {
|
||||||
if (chip->shutdown)
|
if (chip->shutdown)
|
||||||
break;
|
break;
|
||||||
@@ -471,7 +471,7 @@ int snd_usb_mixer_set_ctl_value(struct usb_mixer_elem_info *cval,
|
|||||||
err = -EINVAL;
|
err = -EINVAL;
|
||||||
|
|
||||||
out:
|
out:
|
||||||
mutex_unlock(&chip->shutdown_mutex);
|
up_read(&chip->shutdown_rwsem);
|
||||||
snd_usb_autosuspend(chip);
|
snd_usb_autosuspend(chip);
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
@@ -503,12 +503,12 @@ static int snd_usb_hw_params(struct snd_pcm_substream *substream,
|
|||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
mutex_lock(&subs->stream->chip->shutdown_mutex);
|
down_read(&subs->stream->chip->shutdown_rwsem);
|
||||||
if (subs->stream->chip->shutdown)
|
if (subs->stream->chip->shutdown)
|
||||||
ret = -ENODEV;
|
ret = -ENODEV;
|
||||||
else
|
else
|
||||||
ret = set_format(subs, fmt);
|
ret = set_format(subs, fmt);
|
||||||
mutex_unlock(&subs->stream->chip->shutdown_mutex);
|
up_read(&subs->stream->chip->shutdown_rwsem);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
@@ -531,12 +531,12 @@ static int snd_usb_hw_free(struct snd_pcm_substream *substream)
|
|||||||
subs->cur_audiofmt = NULL;
|
subs->cur_audiofmt = NULL;
|
||||||
subs->cur_rate = 0;
|
subs->cur_rate = 0;
|
||||||
subs->period_bytes = 0;
|
subs->period_bytes = 0;
|
||||||
mutex_lock(&subs->stream->chip->shutdown_mutex);
|
down_read(&subs->stream->chip->shutdown_rwsem);
|
||||||
if (!subs->stream->chip->shutdown) {
|
if (!subs->stream->chip->shutdown) {
|
||||||
stop_endpoints(subs, 0, 1, 1);
|
stop_endpoints(subs, 0, 1, 1);
|
||||||
deactivate_endpoints(subs);
|
deactivate_endpoints(subs);
|
||||||
}
|
}
|
||||||
mutex_unlock(&subs->stream->chip->shutdown_mutex);
|
up_read(&subs->stream->chip->shutdown_rwsem);
|
||||||
return snd_pcm_lib_free_vmalloc_buffer(substream);
|
return snd_pcm_lib_free_vmalloc_buffer(substream);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -558,7 +558,7 @@ static int snd_usb_pcm_prepare(struct snd_pcm_substream *substream)
|
|||||||
return -ENXIO;
|
return -ENXIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
mutex_lock(&subs->stream->chip->shutdown_mutex);
|
down_read(&subs->stream->chip->shutdown_rwsem);
|
||||||
if (subs->stream->chip->shutdown) {
|
if (subs->stream->chip->shutdown) {
|
||||||
ret = -ENODEV;
|
ret = -ENODEV;
|
||||||
goto unlock;
|
goto unlock;
|
||||||
@@ -608,7 +608,7 @@ static int snd_usb_pcm_prepare(struct snd_pcm_substream *substream)
|
|||||||
ret = start_endpoints(subs, 1);
|
ret = start_endpoints(subs, 1);
|
||||||
|
|
||||||
unlock:
|
unlock:
|
||||||
mutex_unlock(&subs->stream->chip->shutdown_mutex);
|
up_read(&subs->stream->chip->shutdown_rwsem);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -37,7 +37,7 @@ struct snd_usb_audio {
|
|||||||
struct usb_interface *pm_intf;
|
struct usb_interface *pm_intf;
|
||||||
u32 usb_id;
|
u32 usb_id;
|
||||||
struct mutex mutex;
|
struct mutex mutex;
|
||||||
struct mutex shutdown_mutex;
|
struct rw_semaphore shutdown_rwsem;
|
||||||
unsigned int shutdown:1;
|
unsigned int shutdown:1;
|
||||||
unsigned int probing:1;
|
unsigned int probing:1;
|
||||||
unsigned int autosuspended:1;
|
unsigned int autosuspended:1;
|
||||||
|
Reference in New Issue
Block a user