summaryrefslogtreecommitdiff
path: root/linux/drivers/staging
diff options
context:
space:
mode:
authorMauro Carvalho Chehab <mchehab@redhat.com>2008-01-10 18:27:26 -0200
committerMauro Carvalho Chehab <mchehab@redhat.com>2008-01-10 18:27:26 -0200
commitb9ed676a5bc4e433cc676cedd0691665cece037f (patch)
tree9f7b3a84e2df7e78e4abe46fbb31bf0c858d9cdf /linux/drivers/staging
parent40d4df5e600eef2f57205c0553682deccc1aa751 (diff)
downloadmediapointer-dvb-s2-b9ed676a5bc4e433cc676cedd0691665cece037f.tar.gz
mediapointer-dvb-s2-b9ed676a5bc4e433cc676cedd0691665cece037f.tar.bz2
tm6000: several changes on usb preparation trying to avoid -EPROTO errors
From: Mauro Carvalho Chehab <mchehab@redhat.com> Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
Diffstat (limited to 'linux/drivers/staging')
-rw-r--r--linux/drivers/staging/tm6000/tm6000-video.c95
1 files changed, 52 insertions, 43 deletions
diff --git a/linux/drivers/staging/tm6000/tm6000-video.c b/linux/drivers/staging/tm6000/tm6000-video.c
index 007db3d84..58f5f55e0 100644
--- a/linux/drivers/staging/tm6000/tm6000-video.c
+++ b/linux/drivers/staging/tm6000/tm6000-video.c
@@ -52,6 +52,8 @@
#define TM6000_MIN_BUF 4
#define TM6000_DEF_BUF 8
+#define TM6000_MAX_ISO_PACKETS 40 /* Max number of ISO packets */
+
/* Declare static vars that will be used as parameters */
static unsigned int vid_limit = 16; /* Video memory limit, in Mb */
static int video_nr = -1; /* /dev/videoN, -1 for autodetect */
@@ -545,7 +547,7 @@ static void tm6000_irq_callback(struct urb *urb)
struct tm6000_buffer *buf;
struct tm6000_dmaqueue *dma_q = urb->context;
struct tm6000_core *dev = container_of(dma_q, struct tm6000_core, vidq);
- int rc, i;
+ int rc;
unsigned long flags;
spin_lock_irqsave(&dev->slock, flags);
@@ -563,11 +565,13 @@ static void tm6000_irq_callback(struct urb *urb)
dev->isoc_ctl.buf = buf;
ret:
+#if 0
/* Reset urb buffers */
for (i = 0; i < urb->number_of_packets; i++) {
urb->iso_frame_desc[i].status = 0;
urb->iso_frame_desc[i].actual_length = 0;
}
+#endif
urb->status = usb_submit_urb(urb, GFP_ATOMIC);
if (urb->status)
@@ -589,8 +593,8 @@ static void tm6000_uninit_isoc(struct tm6000_core *dev)
struct urb *urb;
int i;
- dev->isoc_ctl.nfields=-1;
- dev->isoc_ctl.buf=NULL;
+ dev->isoc_ctl.nfields = -1;
+ dev->isoc_ctl.buf = NULL;
for (i = 0; i < dev->isoc_ctl.num_bufs; i++) {
urb=dev->isoc_ctl.urb[i];
if (urb) {
@@ -626,38 +630,47 @@ static void tm6000_stop_thread(struct tm6000_dmaqueue *dma_q)
tm6000_uninit_isoc(dev);
}
-
/*
* Allocate URBs and start IRQ
*/
-static int tm6000_prepare_isoc(struct tm6000_core *dev,
- int max_packets, int num_bufs)
+static int tm6000_prepare_isoc(struct tm6000_core *dev, unsigned int framesize)
{
struct tm6000_dmaqueue *dma_q = &dev->vidq;
- int i;
- int sb_size, pipe;
+ int i, j, sb_size, pipe, size, max_packets, num_bufs = 5;
struct urb *urb;
- int j, k;
-
- dprintk(dev, V4L2_DEBUG_QUEUE, "Allocating %dx%d packets"
- " of %d bytes each to handle %u size\n",
- max_packets, num_bufs,
- dev->max_isoc_in, dev->isoc_ctl.max_pkt_size);
-
/* De-allocates all pending stuff */
tm6000_uninit_isoc(dev);
- dev->isoc_ctl.num_bufs=num_bufs;
- dev->isoc_ctl.urb=kmalloc(sizeof(void *)*num_bufs,
- GFP_KERNEL);
+ pipe = usb_rcvisocpipe(dev->udev,
+ dev->isoc_in->desc.bEndpointAddress &
+ USB_ENDPOINT_NUMBER_MASK);
+
+ size = usb_maxpacket(dev->udev, pipe, usb_pipeout(pipe));
+
+ if (size > dev->max_isoc_in)
+ size = dev->max_isoc_in;
+
+ dev->isoc_ctl.max_pkt_size = size;
+
+ max_packets = ( framesize + size - 1) / size;
+
+ if (max_packets > TM6000_MAX_ISO_PACKETS)
+ max_packets = TM6000_MAX_ISO_PACKETS;
+
+ sb_size = max_packets * size;
+
+ dev->isoc_ctl.num_bufs = num_bufs;
+
+ dev->isoc_ctl.urb = kmalloc(sizeof(void *)*num_bufs,
+ GFP_KERNEL);
if (!dev->isoc_ctl.urb) {
tm6000_err("cannot alloc memory for usb buffers\n");
return -ENOMEM;
}
- dev->isoc_ctl.transfer_buffer=kmalloc(sizeof(void *)*num_bufs,
+ dev->isoc_ctl.transfer_buffer = kmalloc(sizeof(void *)*num_bufs,
GFP_KERNEL);
if (!dev->isoc_ctl.urb) {
tm6000_err("cannot allocate memory for usbtransfer\n");
@@ -665,12 +678,13 @@ static int tm6000_prepare_isoc(struct tm6000_core *dev,
return -ENOMEM;
}
- dev->isoc_ctl.max_pkt_size=dev->max_isoc_in;
-
- sb_size = max_packets * dev->isoc_ctl.max_pkt_size;
+ dprintk(dev, V4L2_DEBUG_QUEUE, "Allocating %d x %d packets"
+ " (%d bytes) of %d bytes each to handle %u size\n",
+ max_packets, num_bufs, sb_size,
+ dev->max_isoc_in, size);
#if 0
-/* reset streaming vars */
+ /* reset streaming vars */
dev->isoc_ctl.frame_current = NULL;
dev->isoc_ctl.frame_count = 0;
#endif
@@ -698,23 +712,17 @@ static int tm6000_prepare_isoc(struct tm6000_core *dev,
}
memset(dev->isoc_ctl.transfer_buffer[i], 0, sb_size);
- pipe=usb_rcvisocpipe(dev->udev,
- dev->isoc_in->desc.bEndpointAddress &
- USB_ENDPOINT_NUMBER_MASK);
- usb_fill_int_urb(urb, dev->udev, pipe,
- dev->isoc_ctl.transfer_buffer[i],sb_size,
- tm6000_irq_callback, dma_q,
- dev->isoc_in->desc.bInterval);
-
+ usb_fill_bulk_urb(urb, dev->udev, pipe,
+ dev->isoc_ctl.transfer_buffer[i], sb_size,
+ tm6000_irq_callback, dma_q);
+// urb->interval = dev->isoc_in->desc.bInterval;
+ urb->interval = 2;
urb->number_of_packets = max_packets;
- urb->transfer_flags = URB_ISO_ASAP;
+ urb->transfer_flags = URB_ISO_ASAP | URB_NO_TRANSFER_DMA_MAP;
- k = 0;
for (j = 0; j < max_packets; j++) {
- urb->iso_frame_desc[j].offset = k;
- urb->iso_frame_desc[j].length =
- dev->isoc_ctl.max_pkt_size;
- k += dev->isoc_ctl.max_pkt_size;
+ urb->iso_frame_desc[j].offset = size * j;
+ urb->iso_frame_desc[j].length = size;
}
}
@@ -910,17 +918,18 @@ buffer_prepare(struct videobuf_queue *vq, struct videobuf_buffer *vb,
}
if (VIDEOBUF_NEEDS_INIT == buf->vb.state) {
- if (0 != (rc = videobuf_iolock(vq,&buf->vb,NULL)))
+ if (0 != (rc = videobuf_iolock(vq, &buf->vb, NULL)))
goto fail;
- urb_init=1;
+ urb_init = 1;
}
if (!dev->isoc_ctl.num_bufs)
- urb_init=1;
+ urb_init = 1;
if (urb_init) {
- rc = tm6000_prepare_isoc(dev, 128, 1);
- if (rc<0)
+ rc = tm6000_prepare_isoc(dev, buf->vb.size);
+
+ if (rc < 0)
goto fail;
}
@@ -928,7 +937,7 @@ buffer_prepare(struct videobuf_queue *vq, struct videobuf_buffer *vb,
return 0;
fail:
- free_buffer(vq,buf);
+ free_buffer(vq, buf);
return rc;
}