aboutsummaryrefslogtreecommitdiffstats
path: root/sound/usb/usx2y/usbusx2yaudio.c
diff options
context:
space:
mode:
Diffstat (limited to 'sound/usb/usx2y/usbusx2yaudio.c')
-rw-r--r--sound/usb/usx2y/usbusx2yaudio.c184
1 files changed, 87 insertions, 97 deletions
diff --git a/sound/usb/usx2y/usbusx2yaudio.c b/sound/usb/usx2y/usbusx2yaudio.c
index c5989cb7db3f..e1dbfbba8fa9 100644
--- a/sound/usb/usx2y/usbusx2yaudio.c
+++ b/sound/usb/usx2y/usbusx2yaudio.c
@@ -205,47 +205,42 @@ static int usX2Y_urb_submit(snd_usX2Y_substream_t *subs, struct urb *urb, int fr
205static inline int usX2Y_usbframe_complete(snd_usX2Y_substream_t *capsubs, snd_usX2Y_substream_t *playbacksubs, int frame) 205static inline int usX2Y_usbframe_complete(snd_usX2Y_substream_t *capsubs, snd_usX2Y_substream_t *playbacksubs, int frame)
206{ 206{
207 int err, state; 207 int err, state;
208 { 208 struct urb *urb = playbacksubs->completed_urb;
209 struct urb *urb = playbacksubs->completed_urb; 209
210 210 state = atomic_read(&playbacksubs->state);
211 state = atomic_read(&playbacksubs->state); 211 if (NULL != urb) {
212 if (NULL != urb) { 212 if (state == state_RUNNING)
213 if (state == state_RUNNING) 213 usX2Y_urb_play_retire(playbacksubs, urb);
214 usX2Y_urb_play_retire(playbacksubs, urb); 214 else
215 else 215 if (state >= state_PRERUNNING)
216 if (state >= state_PRERUNNING) {
217 atomic_inc(&playbacksubs->state);
218 }
219 } else {
220 switch (state) {
221 case state_STARTING1:
222 urb = playbacksubs->urb[0];
223 atomic_inc(&playbacksubs->state); 216 atomic_inc(&playbacksubs->state);
224 break; 217 } else {
225 case state_STARTING2: 218 switch (state) {
226 urb = playbacksubs->urb[1]; 219 case state_STARTING1:
227 atomic_inc(&playbacksubs->state); 220 urb = playbacksubs->urb[0];
228 break; 221 atomic_inc(&playbacksubs->state);
229 } 222 break;
230 } 223 case state_STARTING2:
231 if (urb) { 224 urb = playbacksubs->urb[1];
232 if ((err = usX2Y_urb_play_prepare(playbacksubs, capsubs->completed_urb, urb)) || 225 atomic_inc(&playbacksubs->state);
233 (err = usX2Y_urb_submit(playbacksubs, urb, frame))) { 226 break;
234 return err;
235 }
236 } 227 }
237
238 playbacksubs->completed_urb = NULL;
239 } 228 }
229 if (urb) {
230 if ((err = usX2Y_urb_play_prepare(playbacksubs, capsubs->completed_urb, urb)) ||
231 (err = usX2Y_urb_submit(playbacksubs, urb, frame)))
232 return err;
233 }
234
235 playbacksubs->completed_urb = NULL;
236
240 state = atomic_read(&capsubs->state); 237 state = atomic_read(&capsubs->state);
241 if (state >= state_PREPARED) { 238 if (state >= state_PREPARED) {
242 if (state == state_RUNNING) { 239 if (state == state_RUNNING) {
243 if ((err = usX2Y_urb_capt_retire(capsubs))) 240 if ((err = usX2Y_urb_capt_retire(capsubs)))
244 return err; 241 return err;
245 } else 242 } else if (state >= state_PRERUNNING)
246 if (state >= state_PRERUNNING) { 243 atomic_inc(&capsubs->state);
247 atomic_inc(&capsubs->state);
248 }
249 if ((err = usX2Y_urb_submit(capsubs, capsubs->completed_urb, frame))) 244 if ((err = usX2Y_urb_submit(capsubs, capsubs->completed_urb, frame)))
250 return err; 245 return err;
251 } 246 }
@@ -429,7 +424,7 @@ static int usX2Y_urbs_allocate(snd_usX2Y_substream_t *subs)
429 } 424 }
430 /* allocate and initialize data urbs */ 425 /* allocate and initialize data urbs */
431 for (i = 0; i < NRURBS; i++) { 426 for (i = 0; i < NRURBS; i++) {
432 struct urb** purb = subs->urb + i; 427 struct urb **purb = subs->urb + i;
433 if (*purb) { 428 if (*purb) {
434 usb_kill_urb(*purb); 429 usb_kill_urb(*purb);
435 continue; 430 continue;
@@ -480,47 +475,45 @@ static int usX2Y_urbs_start(snd_usX2Y_substream_t *subs)
480 goto start; 475 goto start;
481 } 476 }
482 usX2Y->wait_iso_frame = -1; 477 usX2Y->wait_iso_frame = -1;
478
483 start: 479 start:
484 { 480 usX2Y_subs_startup(subs);
485 usX2Y_subs_startup(subs); 481 for (i = 0; i < NRURBS; i++) {
486 for (i = 0; i < NRURBS; i++) { 482 struct urb *urb = subs->urb[i];
487 struct urb *urb = subs->urb[i]; 483 if (usb_pipein(urb->pipe)) {
488 if (usb_pipein(urb->pipe)) { 484 unsigned long pack;
489 unsigned long pack; 485 if (0 == i)
490 if (0 == i) 486 atomic_set(&subs->state, state_STARTING3);
491 atomic_set(&subs->state, state_STARTING3); 487 urb->dev = usX2Y->chip.dev;
492 urb->dev = usX2Y->chip.dev; 488 urb->transfer_flags = URB_ISO_ASAP;
493 urb->transfer_flags = URB_ISO_ASAP; 489 for (pack = 0; pack < nr_of_packs(); pack++) {
494 for (pack = 0; pack < nr_of_packs(); pack++) { 490 urb->iso_frame_desc[pack].offset = subs->maxpacksize * pack;
495 urb->iso_frame_desc[pack].offset = subs->maxpacksize * pack; 491 urb->iso_frame_desc[pack].length = subs->maxpacksize;
496 urb->iso_frame_desc[pack].length = subs->maxpacksize; 492 }
497 } 493 urb->transfer_buffer_length = subs->maxpacksize * nr_of_packs();
498 urb->transfer_buffer_length = subs->maxpacksize * nr_of_packs(); 494 if ((err = usb_submit_urb(urb, GFP_ATOMIC)) < 0) {
499 if ((err = usb_submit_urb(urb, GFP_ATOMIC)) < 0) { 495 snd_printk (KERN_ERR "cannot submit datapipe for urb %d, err = %d\n", i, err);
500 snd_printk (KERN_ERR "cannot submit datapipe for urb %d, err = %d\n", i, err); 496 err = -EPIPE;
501 err = -EPIPE; 497 goto cleanup;
502 goto cleanup;
503 } else {
504 if (0 > usX2Y->wait_iso_frame)
505 usX2Y->wait_iso_frame = urb->start_frame;
506 }
507 urb->transfer_flags = 0;
508 } else { 498 } else {
509 atomic_set(&subs->state, state_STARTING1); 499 if (0 > usX2Y->wait_iso_frame)
510 break; 500 usX2Y->wait_iso_frame = urb->start_frame;
511 } 501 }
502 urb->transfer_flags = 0;
503 } else {
504 atomic_set(&subs->state, state_STARTING1);
505 break;
512 } 506 }
513 err = 0; 507 }
514 wait_event(usX2Y->prepare_wait_queue, NULL == usX2Y->prepare_subs); 508 err = 0;
515 if (atomic_read(&subs->state) != state_PREPARED) { 509 wait_event(usX2Y->prepare_wait_queue, NULL == usX2Y->prepare_subs);
516 err = -EPIPE; 510 if (atomic_read(&subs->state) != state_PREPARED)
517 } 511 err = -EPIPE;
518 512
519 cleanup: 513 cleanup:
520 if (err) { 514 if (err) {
521 usX2Y_subs_startup_finish(usX2Y); 515 usX2Y_subs_startup_finish(usX2Y);
522 usX2Y_clients_stop(usX2Y); // something is completely wroong > stop evrything 516 usX2Y_clients_stop(usX2Y); // something is completely wroong > stop evrything
523 }
524 } 517 }
525 return err; 518 return err;
526} 519}
@@ -667,13 +660,12 @@ static int usX2Y_rate_set(usX2Ydev_t *usX2Y, int rate)
667 struct s_c2 *ra = rate == 48000 ? SetRate48000 : SetRate44100; 660 struct s_c2 *ra = rate == 48000 ? SetRate48000 : SetRate44100;
668 661
669 if (usX2Y->rate != rate) { 662 if (usX2Y->rate != rate) {
670 us = kmalloc(sizeof(*us) + sizeof(struct urb*) * NOOF_SETRATE_URBS, GFP_KERNEL); 663 us = kzalloc(sizeof(*us) + sizeof(struct urb*) * NOOF_SETRATE_URBS, GFP_KERNEL);
671 if (NULL == us) { 664 if (NULL == us) {
672 err = -ENOMEM; 665 err = -ENOMEM;
673 goto cleanup; 666 goto cleanup;
674 } 667 }
675 memset(us, 0, sizeof(*us) + sizeof(struct urb*) * NOOF_SETRATE_URBS); 668 usbdata = kmalloc(sizeof(int) * NOOF_SETRATE_URBS, GFP_KERNEL);
676 usbdata = kmalloc(sizeof(int)*NOOF_SETRATE_URBS, GFP_KERNEL);
677 if (NULL == usbdata) { 669 if (NULL == usbdata) {
678 err = -ENOMEM; 670 err = -ENOMEM;
679 goto cleanup; 671 goto cleanup;
@@ -713,9 +705,8 @@ static int usX2Y_rate_set(usX2Ydev_t *usX2Y, int rate)
713 usX2Y->US04 = NULL; 705 usX2Y->US04 = NULL;
714 kfree(usbdata); 706 kfree(usbdata);
715 kfree(us); 707 kfree(us);
716 if (!err) { 708 if (!err)
717 usX2Y->rate = rate; 709 usX2Y->rate = rate;
718 }
719 } 710 }
720 } 711 }
721 712
@@ -759,30 +750,29 @@ static int snd_usX2Y_pcm_hw_params(snd_pcm_substream_t *substream,
759 int err = 0; 750 int err = 0;
760 unsigned int rate = params_rate(hw_params); 751 unsigned int rate = params_rate(hw_params);
761 snd_pcm_format_t format = params_format(hw_params); 752 snd_pcm_format_t format = params_format(hw_params);
762 snd_printdd("snd_usX2Y_hw_params(%p, %p)\n", substream, hw_params);
763
764 { // all pcm substreams off one usX2Y have to operate at the same rate & format
765 snd_card_t *card = substream->pstr->pcm->card; 753 snd_card_t *card = substream->pstr->pcm->card;
766 struct list_head *list; 754 struct list_head *list;
767 list_for_each(list, &card->devices) { 755
768 snd_device_t *dev; 756 snd_printdd("snd_usX2Y_hw_params(%p, %p)\n", substream, hw_params);
769 snd_pcm_t *pcm; 757 // all pcm substreams off one usX2Y have to operate at the same rate & format
770 int s; 758 list_for_each(list, &card->devices) {
771 dev = snd_device(list); 759 snd_device_t *dev;
772 if (dev->type != SNDRV_DEV_PCM) 760 snd_pcm_t *pcm;
773 continue; 761 int s;
774 pcm = dev->device_data; 762 dev = snd_device(list);
775 for (s = 0; s < 2; ++s) { 763 if (dev->type != SNDRV_DEV_PCM)
776 snd_pcm_substream_t *test_substream; 764 continue;
777 test_substream = pcm->streams[s].substream; 765 pcm = dev->device_data;
778 if (test_substream && test_substream != substream && 766 for (s = 0; s < 2; ++s) {
779 test_substream->runtime && 767 snd_pcm_substream_t *test_substream;
780 ((test_substream->runtime->format && 768 test_substream = pcm->streams[s].substream;
781 test_substream->runtime->format != format) || 769 if (test_substream && test_substream != substream &&
782 (test_substream->runtime->rate && 770 test_substream->runtime &&
783 test_substream->runtime->rate != rate))) 771 ((test_substream->runtime->format &&
784 return -EINVAL; 772 test_substream->runtime->format != format) ||
785 } 773 (test_substream->runtime->rate &&
774 test_substream->runtime->rate != rate)))
775 return -EINVAL;
786 } 776 }
787 } 777 }
788 if (0 > (err = snd_pcm_lib_malloc_pages(substream, params_buffer_bytes(hw_params)))) { 778 if (0 > (err = snd_pcm_lib_malloc_pages(substream, params_buffer_bytes(hw_params)))) {