aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMing Lei <tom.leiming@gmail.com>2008-09-16 02:32:20 -0400
committerMauro Carvalho Chehab <mchehab@redhat.com>2008-10-12 07:37:08 -0400
commita31a4055473bf0a7b2b06cb2262347200d0711e1 (patch)
tree296667c25e67acd933c55c71b85b619b5918ae5e
parentd63beb9ef004ff9587b3c466361276254d57d7a7 (diff)
V4L/DVB:usbvideo:don't use part of buffer for USB transfer #4
The status[] is part of uvc_device structure. We can't make sure the address of status is at a cache-line boundary in all archs,so status[] might share a cache-line with some fields in uvc_structure. This can lead to some cache coherence issues(http://lwn.net/Articles/2265/). Use dynamically allocated buffer instead. Signed-off-by: Ming Lei <tom.leiming@gmail.com> Acked-by: Laurent Pinchart <laurent.pinchart@skynet.be> Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
-rw-r--r--drivers/media/video/uvc/uvc_status.c11
-rw-r--r--drivers/media/video/uvc/uvcvideo.h4
2 files changed, 12 insertions, 3 deletions
diff --git a/drivers/media/video/uvc/uvc_status.c b/drivers/media/video/uvc/uvc_status.c
index 75e678ac54eb..5d60b264d59a 100644
--- a/drivers/media/video/uvc/uvc_status.c
+++ b/drivers/media/video/uvc/uvc_status.c
@@ -177,9 +177,15 @@ int uvc_status_init(struct uvc_device *dev)
177 177
178 uvc_input_init(dev); 178 uvc_input_init(dev);
179 179
180 dev->status = kzalloc(UVC_MAX_STATUS_SIZE, GFP_KERNEL);
181 if (dev->status == NULL)
182 return -ENOMEM;
183
180 dev->int_urb = usb_alloc_urb(0, GFP_KERNEL); 184 dev->int_urb = usb_alloc_urb(0, GFP_KERNEL);
181 if (dev->int_urb == NULL) 185 if (dev->int_urb == NULL) {
186 kfree(dev->status);
182 return -ENOMEM; 187 return -ENOMEM;
188 }
183 189
184 pipe = usb_rcvintpipe(dev->udev, ep->desc.bEndpointAddress); 190 pipe = usb_rcvintpipe(dev->udev, ep->desc.bEndpointAddress);
185 191
@@ -192,7 +198,7 @@ int uvc_status_init(struct uvc_device *dev)
192 interval = fls(interval) - 1; 198 interval = fls(interval) - 1;
193 199
194 usb_fill_int_urb(dev->int_urb, dev->udev, pipe, 200 usb_fill_int_urb(dev->int_urb, dev->udev, pipe,
195 dev->status, sizeof dev->status, uvc_status_complete, 201 dev->status, UVC_MAX_STATUS_SIZE, uvc_status_complete,
196 dev, interval); 202 dev, interval);
197 203
198 return usb_submit_urb(dev->int_urb, GFP_KERNEL); 204 return usb_submit_urb(dev->int_urb, GFP_KERNEL);
@@ -202,6 +208,7 @@ void uvc_status_cleanup(struct uvc_device *dev)
202{ 208{
203 usb_kill_urb(dev->int_urb); 209 usb_kill_urb(dev->int_urb);
204 usb_free_urb(dev->int_urb); 210 usb_free_urb(dev->int_urb);
211 kfree(dev->status);
205 uvc_input_cleanup(dev); 212 uvc_input_cleanup(dev);
206} 213}
207 214
diff --git a/drivers/media/video/uvc/uvcvideo.h b/drivers/media/video/uvc/uvcvideo.h
index bafe3406e305..9a6bc1aafb16 100644
--- a/drivers/media/video/uvc/uvcvideo.h
+++ b/drivers/media/video/uvc/uvcvideo.h
@@ -303,6 +303,8 @@ struct uvc_xu_control {
303#define UVC_MAX_FRAME_SIZE (16*1024*1024) 303#define UVC_MAX_FRAME_SIZE (16*1024*1024)
304/* Maximum number of video buffers. */ 304/* Maximum number of video buffers. */
305#define UVC_MAX_VIDEO_BUFFERS 32 305#define UVC_MAX_VIDEO_BUFFERS 32
306/* Maximum status buffer size in bytes of interrupt URB. */
307#define UVC_MAX_STATUS_SIZE 16
306 308
307#define UVC_CTRL_CONTROL_TIMEOUT 300 309#define UVC_CTRL_CONTROL_TIMEOUT 300
308#define UVC_CTRL_STREAMING_TIMEOUT 1000 310#define UVC_CTRL_STREAMING_TIMEOUT 1000
@@ -634,7 +636,7 @@ struct uvc_device {
634 /* Status Interrupt Endpoint */ 636 /* Status Interrupt Endpoint */
635 struct usb_host_endpoint *int_ep; 637 struct usb_host_endpoint *int_ep;
636 struct urb *int_urb; 638 struct urb *int_urb;
637 __u8 status[16]; 639 __u8 *status;
638 struct input_dev *input; 640 struct input_dev *input;
639 641
640 /* Video Streaming interfaces */ 642 /* Video Streaming interfaces */