Commit 18a9079f authored by Jaroslav Kysela's avatar Jaroslav Kysela

ALSA CVS update - Takashi Iwai <tiwai@suse.de>

USB generic driver
fix by Clemens Ladisch <clemens@ladisch.de>:

- don't clear active_mask bits until it's clear that the URB is _not_
  resubmitted, to prevent a race with unlinking
- initialize active_mask and unlink_mask each time before URBs are
  started
- don't call sleeping functions in trigger callback
parent cc4f2363
...@@ -573,21 +573,18 @@ static void snd_complete_urb(struct urb *urb, struct pt_regs *regs) ...@@ -573,21 +573,18 @@ static void snd_complete_urb(struct urb *urb, struct pt_regs *regs)
snd_urb_ctx_t *ctx = (snd_urb_ctx_t *)urb->context; snd_urb_ctx_t *ctx = (snd_urb_ctx_t *)urb->context;
snd_usb_substream_t *subs = ctx->subs; snd_usb_substream_t *subs = ctx->subs;
snd_pcm_substream_t *substream = ctx->subs->pcm_substream; snd_pcm_substream_t *substream = ctx->subs->pcm_substream;
int err; int err = 0;
clear_bit(ctx->index, &subs->active_mask); if ((subs->running && subs->ops.retire(subs, substream->runtime, urb)) ||
clear_bit(ctx->index, &subs->unlink_mask); ! subs->running || /* can be stopped during retire callback */
if (subs->running && subs->ops.retire(subs, substream->runtime, urb)) (err = subs->ops.prepare(subs, substream->runtime, urb)) < 0 ||
return;
if (! subs->running) /* can be stopped during retire callback */
return;
if ((err = subs->ops.prepare(subs, substream->runtime, urb)) < 0 ||
(err = usb_submit_urb(urb, GFP_ATOMIC)) < 0) { (err = usb_submit_urb(urb, GFP_ATOMIC)) < 0) {
snd_printd(KERN_ERR "cannot submit urb (err = %d)\n", err); clear_bit(ctx->index, &subs->active_mask);
snd_pcm_stop(substream, SNDRV_PCM_STATE_XRUN); if (err < 0) {
return; snd_printd(KERN_ERR "cannot submit urb (err = %d)\n", err);
snd_pcm_stop(substream, SNDRV_PCM_STATE_XRUN);
}
} }
set_bit(ctx->index, &subs->active_mask);
} }
...@@ -599,21 +596,18 @@ static void snd_complete_sync_urb(struct urb *urb, struct pt_regs *regs) ...@@ -599,21 +596,18 @@ static void snd_complete_sync_urb(struct urb *urb, struct pt_regs *regs)
snd_urb_ctx_t *ctx = (snd_urb_ctx_t *)urb->context; snd_urb_ctx_t *ctx = (snd_urb_ctx_t *)urb->context;
snd_usb_substream_t *subs = ctx->subs; snd_usb_substream_t *subs = ctx->subs;
snd_pcm_substream_t *substream = ctx->subs->pcm_substream; snd_pcm_substream_t *substream = ctx->subs->pcm_substream;
int err; int err = 0;
clear_bit(ctx->index + 16, &subs->active_mask); if ((subs->running && subs->ops.retire_sync(subs, substream->runtime, urb)) ||
clear_bit(ctx->index + 16, &subs->unlink_mask); ! subs->running || /* can be stopped during retire callback */
if (subs->running && subs->ops.retire_sync(subs, substream->runtime, urb)) (err = subs->ops.prepare_sync(subs, substream->runtime, urb)) < 0 ||
return;
if (! subs->running) /* can be stopped during retire callback */
return;
if ((err = subs->ops.prepare_sync(subs, substream->runtime, urb)) < 0 ||
(err = usb_submit_urb(urb, GFP_ATOMIC)) < 0) { (err = usb_submit_urb(urb, GFP_ATOMIC)) < 0) {
snd_printd(KERN_ERR "cannot submit sync urb (err = %d)\n", err); clear_bit(ctx->index + 16, &subs->active_mask);
snd_pcm_stop(substream, SNDRV_PCM_STATE_XRUN); if (err < 0) {
return; snd_printd(KERN_ERR "cannot submit sync urb (err = %d)\n", err);
snd_pcm_stop(substream, SNDRV_PCM_STATE_XRUN);
}
} }
set_bit(ctx->index + 16, &subs->active_mask);
} }
...@@ -694,9 +688,11 @@ static int start_urbs(snd_usb_substream_t *subs, snd_pcm_runtime_t *runtime) ...@@ -694,9 +688,11 @@ static int start_urbs(snd_usb_substream_t *subs, snd_pcm_runtime_t *runtime)
} }
} }
subs->active_mask = 0;
subs->unlink_mask = 0;
subs->running = 1; subs->running = 1;
for (i = 0; i < subs->nurbs; i++) { for (i = 0; i < subs->nurbs; i++) {
if ((err = usb_submit_urb(subs->dataurb[i].urb, GFP_KERNEL)) < 0) { if ((err = usb_submit_urb(subs->dataurb[i].urb, GFP_ATOMIC)) < 0) {
snd_printk(KERN_ERR "cannot submit datapipe for urb %d, err = %d\n", i, err); snd_printk(KERN_ERR "cannot submit datapipe for urb %d, err = %d\n", i, err);
goto __error; goto __error;
} }
...@@ -704,7 +700,7 @@ static int start_urbs(snd_usb_substream_t *subs, snd_pcm_runtime_t *runtime) ...@@ -704,7 +700,7 @@ static int start_urbs(snd_usb_substream_t *subs, snd_pcm_runtime_t *runtime)
} }
if (subs->syncpipe) { if (subs->syncpipe) {
for (i = 0; i < SYNC_URBS; i++) { for (i = 0; i < SYNC_URBS; i++) {
if ((err = usb_submit_urb(subs->syncurb[i].urb, GFP_KERNEL)) < 0) { if ((err = usb_submit_urb(subs->syncurb[i].urb, GFP_ATOMIC)) < 0) {
snd_printk(KERN_ERR "cannot submit syncpipe for urb %d, err = %d\n", i, err); snd_printk(KERN_ERR "cannot submit syncpipe for urb %d, err = %d\n", i, err);
goto __error; goto __error;
} }
...@@ -843,8 +839,6 @@ static int init_substream_urbs(snd_usb_substream_t *subs, unsigned int period_by ...@@ -843,8 +839,6 @@ static int init_substream_urbs(snd_usb_substream_t *subs, unsigned int period_by
subs->hwptr_done = 0; subs->hwptr_done = 0;
subs->transfer_sched = 0; subs->transfer_sched = 0;
subs->transfer_done = 0; subs->transfer_done = 0;
subs->active_mask = 0;
subs->unlink_mask = 0;
/* calculate the max. size of packet */ /* calculate the max. size of packet */
maxsize = ((subs->freqmax + 0x3fff) * (frame_bits >> 3)) >> 14; maxsize = ((subs->freqmax + 0x3fff) * (frame_bits >> 3)) >> 14;
...@@ -1283,9 +1277,6 @@ static int snd_usb_pcm_prepare(snd_pcm_substream_t *substream) ...@@ -1283,9 +1277,6 @@ static int snd_usb_pcm_prepare(snd_pcm_substream_t *substream)
subs->maxframesize = bytes_to_frames(runtime, subs->maxpacksize); subs->maxframesize = bytes_to_frames(runtime, subs->maxpacksize);
subs->curframesize = bytes_to_frames(runtime, subs->curpacksize); subs->curframesize = bytes_to_frames(runtime, subs->curpacksize);
/* deactivate urbs to be sure */
deactivate_urbs(subs, 0, 0);
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