dvb-usb-urb.c 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211
  1. /* dvb-usb-urb.c is part of the DVB USB library.
  2. *
  3. * Copyright (C) 2004-5 Patrick Boettcher (patrick.boettcher@desy.de)
  4. * see dvb-usb-init.c for copyright information.
  5. *
  6. * This file contains functions for initializing and handling the
  7. * USB and URB stuff.
  8. */
  9. #include "dvb-usb-common.h"
  10. int dvb_usb_generic_rw(struct dvb_usb_device *d, u8 *wbuf, u16 wlen, u8 *rbuf,
  11. u16 rlen, int delay_ms)
  12. {
  13. int actlen,ret = -ENOMEM;
  14. if (d->props.generic_bulk_ctrl_endpoint == 0) {
  15. err("endpoint for generic control not specified.");
  16. return -EINVAL;
  17. }
  18. if (wbuf == NULL || wlen == 0)
  19. return -EINVAL;
  20. if ((ret = down_interruptible(&d->usb_sem)))
  21. return ret;
  22. debug_dump(wbuf,wlen,deb_xfer);
  23. ret = usb_bulk_msg(d->udev,usb_sndbulkpipe(d->udev,
  24. d->props.generic_bulk_ctrl_endpoint), wbuf,wlen,&actlen,
  25. 2*HZ);
  26. if (ret)
  27. err("bulk message failed: %d (%d/%d)",ret,wlen,actlen);
  28. else
  29. ret = actlen != wlen ? -1 : 0;
  30. /* an answer is expected, and no error before */
  31. if (!ret && rbuf && rlen) {
  32. if (delay_ms)
  33. msleep(delay_ms);
  34. ret = usb_bulk_msg(d->udev,usb_rcvbulkpipe(d->udev,
  35. d->props.generic_bulk_ctrl_endpoint),rbuf,rlen,&actlen,
  36. 2*HZ);
  37. if (ret)
  38. err("recv bulk message failed: %d",ret);
  39. else
  40. debug_dump(rbuf,actlen,deb_xfer);
  41. }
  42. up(&d->usb_sem);
  43. return ret;
  44. }
  45. EXPORT_SYMBOL(dvb_usb_generic_rw);
  46. int dvb_usb_generic_write(struct dvb_usb_device *d, u8 *buf, u16 len)
  47. {
  48. return dvb_usb_generic_rw(d,buf,len,NULL,0,0);
  49. }
  50. EXPORT_SYMBOL(dvb_usb_generic_write);
  51. static void dvb_usb_bulk_urb_complete(struct urb *urb, struct pt_regs *ptregs)
  52. {
  53. struct dvb_usb_device *d = urb->context;
  54. deb_ts("bulk urb completed. feedcount: %d, status: %d, length: %d\n",d->feedcount,urb->status,
  55. urb->actual_length);
  56. switch (urb->status) {
  57. case 0: /* success */
  58. case -ETIMEDOUT: /* NAK */
  59. break;
  60. case -ECONNRESET: /* kill */
  61. case -ENOENT:
  62. case -ESHUTDOWN:
  63. return;
  64. default: /* error */
  65. deb_ts("urb completition error %d.", urb->status);
  66. break;
  67. }
  68. if (d->feedcount > 0 && urb->actual_length > 0) {
  69. if (d->state & DVB_USB_STATE_DVB)
  70. dvb_dmx_swfilter(&d->demux, (u8*) urb->transfer_buffer,urb->actual_length);
  71. } else
  72. deb_ts("URB dropped because of feedcount.\n");
  73. usb_submit_urb(urb,GFP_ATOMIC);
  74. }
  75. int dvb_usb_urb_kill(struct dvb_usb_device *d)
  76. {
  77. int i;
  78. for (i = 0; i < d->urbs_submitted; i++) {
  79. deb_info("killing URB no. %d.\n",i);
  80. /* stop the URB */
  81. usb_kill_urb(d->urb_list[i]);
  82. }
  83. d->urbs_submitted = 0;
  84. return 0;
  85. }
  86. int dvb_usb_urb_submit(struct dvb_usb_device *d)
  87. {
  88. int i,ret;
  89. for (i = 0; i < d->urbs_initialized; i++) {
  90. deb_info("submitting URB no. %d\n",i);
  91. if ((ret = usb_submit_urb(d->urb_list[i],GFP_ATOMIC))) {
  92. err("could not submit URB no. %d - get them all back\n",i);
  93. dvb_usb_urb_kill(d);
  94. return ret;
  95. }
  96. d->urbs_submitted++;
  97. }
  98. return 0;
  99. }
  100. static int dvb_usb_bulk_urb_init(struct dvb_usb_device *d)
  101. {
  102. int i,bufsize = d->props.urb.count * d->props.urb.u.bulk.buffersize;
  103. deb_info("allocate %d bytes as buffersize for all URBs\n",bufsize);
  104. /* allocate the actual buffer for the URBs */
  105. if ((d->buffer = usb_buffer_alloc(d->udev, bufsize, SLAB_ATOMIC, &d->dma_handle)) == NULL) {
  106. deb_info("not enough memory for urb-buffer allocation.\n");
  107. return -ENOMEM;
  108. }
  109. deb_info("allocation successful\n");
  110. memset(d->buffer,0,bufsize);
  111. d->state |= DVB_USB_STATE_URB_BUF;
  112. /* allocate the URBs */
  113. for (i = 0; i < d->props.urb.count; i++) {
  114. if (!(d->urb_list[i] = usb_alloc_urb(0,GFP_ATOMIC))) {
  115. return -ENOMEM;
  116. }
  117. usb_fill_bulk_urb( d->urb_list[i], d->udev,
  118. usb_rcvbulkpipe(d->udev,d->props.urb.endpoint),
  119. &d->buffer[i*d->props.urb.u.bulk.buffersize],
  120. d->props.urb.u.bulk.buffersize,
  121. dvb_usb_bulk_urb_complete, d);
  122. d->urb_list[i]->transfer_flags = 0;
  123. d->urbs_initialized++;
  124. }
  125. return 0;
  126. }
  127. int dvb_usb_urb_init(struct dvb_usb_device *d)
  128. {
  129. /*
  130. * when reloading the driver w/o replugging the device
  131. * sometimes a timeout occures, this helps
  132. */
  133. if (d->props.generic_bulk_ctrl_endpoint != 0) {
  134. usb_clear_halt(d->udev,usb_sndbulkpipe(d->udev,d->props.generic_bulk_ctrl_endpoint));
  135. usb_clear_halt(d->udev,usb_rcvbulkpipe(d->udev,d->props.generic_bulk_ctrl_endpoint));
  136. }
  137. usb_clear_halt(d->udev,usb_rcvbulkpipe(d->udev,d->props.urb.endpoint));
  138. /* allocate the array for the data transfer URBs */
  139. d->urb_list = kmalloc(d->props.urb.count * sizeof(struct urb *),GFP_KERNEL);
  140. if (d->urb_list == NULL)
  141. return -ENOMEM;
  142. memset(d->urb_list,0,d->props.urb.count * sizeof(struct urb *));
  143. d->state |= DVB_USB_STATE_URB_LIST;
  144. switch (d->props.urb.type) {
  145. case DVB_USB_BULK:
  146. return dvb_usb_bulk_urb_init(d);
  147. case DVB_USB_ISOC:
  148. err("isochronous transfer not yet implemented in dvb-usb.");
  149. return -EINVAL;
  150. default:
  151. err("unkown URB-type for data transfer.");
  152. return -EINVAL;
  153. }
  154. }
  155. int dvb_usb_urb_exit(struct dvb_usb_device *d)
  156. {
  157. int i;
  158. dvb_usb_urb_kill(d);
  159. if (d->state & DVB_USB_STATE_URB_LIST) {
  160. for (i = 0; i < d->urbs_initialized; i++) {
  161. if (d->urb_list[i] != NULL) {
  162. deb_info("freeing URB no. %d.\n",i);
  163. /* free the URBs */
  164. usb_free_urb(d->urb_list[i]);
  165. }
  166. }
  167. d->urbs_initialized = 0;
  168. /* free the urb array */
  169. kfree(d->urb_list);
  170. d->state &= ~DVB_USB_STATE_URB_LIST;
  171. }
  172. if (d->state & DVB_USB_STATE_URB_BUF)
  173. usb_buffer_free(d->udev, d->props.urb.u.bulk.buffersize * d->props.urb.count,
  174. d->buffer, d->dma_handle);
  175. d->state &= ~DVB_USB_STATE_URB_BUF;
  176. return 0;
  177. }