aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/media/video/usbvision/usbvision-video.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/media/video/usbvision/usbvision-video.c')
-rw-r--r--drivers/media/video/usbvision/usbvision-video.c57
1 files changed, 35 insertions, 22 deletions
diff --git a/drivers/media/video/usbvision/usbvision-video.c b/drivers/media/video/usbvision/usbvision-video.c
index 6a61ebcdf130..cfbfda47ec4f 100644
--- a/drivers/media/video/usbvision/usbvision-video.c
+++ b/drivers/media/video/usbvision/usbvision-video.c
@@ -391,19 +391,14 @@ static int usbvision_v4l2_open(struct inode *inode, struct file *file)
391 if (usbvision->user) 391 if (usbvision->user)
392 errCode = -EBUSY; 392 errCode = -EBUSY;
393 else { 393 else {
394 /* Allocate memory for the frame buffers */ 394 /* Allocate memory for the scratch ring buffer */
395 errCode = usbvision_frames_alloc(usbvision); 395 errCode = usbvision_scratch_alloc(usbvision);
396 if(!errCode) { 396 if (isocMode==ISOC_MODE_COMPRESS) {
397 /* Allocate memory for the scratch ring buffer */ 397 /* Allocate intermediate decompression buffers only if needed */
398 errCode = usbvision_scratch_alloc(usbvision); 398 errCode = usbvision_decompress_alloc(usbvision);
399 if ((!errCode) && (isocMode==ISOC_MODE_COMPRESS)) {
400 /* Allocate intermediate decompression buffers only if needed */
401 errCode = usbvision_decompress_alloc(usbvision);
402 }
403 } 399 }
404 if (errCode) { 400 if (errCode) {
405 /* Deallocate all buffers if trouble */ 401 /* Deallocate all buffers if trouble */
406 usbvision_frames_free(usbvision);
407 usbvision_scratch_free(usbvision); 402 usbvision_scratch_free(usbvision);
408 usbvision_decompress_free(usbvision); 403 usbvision_decompress_free(usbvision);
409 } 404 }
@@ -476,6 +471,7 @@ static int usbvision_v4l2_close(struct inode *inode, struct file *file)
476 471
477 usbvision_decompress_free(usbvision); 472 usbvision_decompress_free(usbvision);
478 usbvision_frames_free(usbvision); 473 usbvision_frames_free(usbvision);
474 usbvision_empty_framequeues(usbvision);
479 usbvision_scratch_free(usbvision); 475 usbvision_scratch_free(usbvision);
480 476
481 usbvision->user--; 477 usbvision->user--;
@@ -809,7 +805,9 @@ static int usbvision_v4l2_do_ioctl(struct inode *inode, struct file *file,
809 return ret; 805 return ret;
810 } 806 }
811 807
808 usbvision_frames_free(usbvision);
812 usbvision_empty_framequeues(usbvision); 809 usbvision_empty_framequeues(usbvision);
810 vr->count = usbvision_frames_alloc(usbvision,vr->count);
813 811
814 usbvision->curFrame = NULL; 812 usbvision->curFrame = NULL;
815 813
@@ -826,7 +824,7 @@ static int usbvision_v4l2_do_ioctl(struct inode *inode, struct file *file,
826 if(vb->type != V4L2_CAP_VIDEO_CAPTURE) { 824 if(vb->type != V4L2_CAP_VIDEO_CAPTURE) {
827 return -EINVAL; 825 return -EINVAL;
828 } 826 }
829 if(vb->index>=USBVISION_NUMFRAMES) { 827 if(vb->index>=usbvision->num_frames) {
830 return -EINVAL; 828 return -EINVAL;
831 } 829 }
832 // Updating the corresponding frame state 830 // Updating the corresponding frame state
@@ -840,7 +838,7 @@ static int usbvision_v4l2_do_ioctl(struct inode *inode, struct file *file,
840 vb->flags |= V4L2_BUF_FLAG_MAPPED; 838 vb->flags |= V4L2_BUF_FLAG_MAPPED;
841 vb->memory = V4L2_MEMORY_MMAP; 839 vb->memory = V4L2_MEMORY_MMAP;
842 840
843 vb->m.offset = vb->index*usbvision->max_frame_size; 841 vb->m.offset = vb->index*PAGE_ALIGN(usbvision->max_frame_size);
844 842
845 vb->memory = V4L2_MEMORY_MMAP; 843 vb->memory = V4L2_MEMORY_MMAP;
846 vb->field = V4L2_FIELD_NONE; 844 vb->field = V4L2_FIELD_NONE;
@@ -859,7 +857,7 @@ static int usbvision_v4l2_do_ioctl(struct inode *inode, struct file *file,
859 if(vb->type != V4L2_CAP_VIDEO_CAPTURE) { 857 if(vb->type != V4L2_CAP_VIDEO_CAPTURE) {
860 return -EINVAL; 858 return -EINVAL;
861 } 859 }
862 if(vb->index>=USBVISION_NUMFRAMES) { 860 if(vb->index>=usbvision->num_frames) {
863 return -EINVAL; 861 return -EINVAL;
864 } 862 }
865 863
@@ -1029,6 +1027,7 @@ static int usbvision_v4l2_do_ioctl(struct inode *inode, struct file *file,
1029 if ((ret = usbvision_stream_interrupt(usbvision))) 1027 if ((ret = usbvision_stream_interrupt(usbvision)))
1030 return ret; 1028 return ret;
1031 } 1029 }
1030 usbvision_frames_free(usbvision);
1032 usbvision_empty_framequeues(usbvision); 1031 usbvision_empty_framequeues(usbvision);
1033 1032
1034 usbvision->curFrame = NULL; 1033 usbvision->curFrame = NULL;
@@ -1075,12 +1074,24 @@ static ssize_t usbvision_v4l2_read(struct file *file, char __user *buf,
1075 if (!USBVISION_IS_OPERATIONAL(usbvision) || (buf == NULL)) 1074 if (!USBVISION_IS_OPERATIONAL(usbvision) || (buf == NULL))
1076 return -EFAULT; 1075 return -EFAULT;
1077 1076
1078 /* no stream is running, make it running ! */ 1077 /* This entry point is compatible with the mmap routines so that a user can do either
1079 usbvision->streaming = Stream_On; 1078 VIDIOC_QBUF/VIDIOC_DQBUF to get frames or call read on the device. */
1080 call_i2c_clients(usbvision,VIDIOC_STREAMON , NULL); 1079 if(!usbvision->num_frames) {
1080 /* First, allocate some frames to work with if this has not been done with
1081 VIDIOC_REQBUF */
1082 usbvision_frames_free(usbvision);
1083 usbvision_empty_framequeues(usbvision);
1084 usbvision_frames_alloc(usbvision,USBVISION_NUMFRAMES);
1085 }
1086
1087 if(usbvision->streaming != Stream_On) {
1088 /* no stream is running, make it running ! */
1089 usbvision->streaming = Stream_On;
1090 call_i2c_clients(usbvision,VIDIOC_STREAMON , NULL);
1091 }
1081 1092
1082 /* First, enqueue as many frames as possible (like a user of VIDIOC_QBUF would do) */ 1093 /* Then, enqueue as many frames as possible (like a user of VIDIOC_QBUF would do) */
1083 for(i=0;i<USBVISION_NUMFRAMES;i++) { 1094 for(i=0;i<usbvision->num_frames;i++) {
1084 frame = &usbvision->frame[i]; 1095 frame = &usbvision->frame[i];
1085 if(frame->grabstate == FrameState_Unused) { 1096 if(frame->grabstate == FrameState_Unused) {
1086 /* Mark it as ready and enqueue frame */ 1097 /* Mark it as ready and enqueue frame */
@@ -1157,6 +1168,8 @@ static int usbvision_v4l2_mmap(struct file *file, struct vm_area_struct *vma)
1157 struct video_device *dev = video_devdata(file); 1168 struct video_device *dev = video_devdata(file);
1158 struct usb_usbvision *usbvision = (struct usb_usbvision *) video_get_drvdata(dev); 1169 struct usb_usbvision *usbvision = (struct usb_usbvision *) video_get_drvdata(dev);
1159 1170
1171 PDEBUG(DBG_MMAP, "mmap");
1172
1160 down(&usbvision->lock); 1173 down(&usbvision->lock);
1161 1174
1162 if (!USBVISION_IS_OPERATIONAL(usbvision)) { 1175 if (!USBVISION_IS_OPERATIONAL(usbvision)) {
@@ -1165,16 +1178,16 @@ static int usbvision_v4l2_mmap(struct file *file, struct vm_area_struct *vma)
1165 } 1178 }
1166 1179
1167 if (!(vma->vm_flags & VM_WRITE) || 1180 if (!(vma->vm_flags & VM_WRITE) ||
1168 size != PAGE_ALIGN(usbvision->curwidth*usbvision->curheight*usbvision->palette.bytes_per_pixel)) { 1181 size != PAGE_ALIGN(usbvision->max_frame_size)) {
1169 up(&usbvision->lock); 1182 up(&usbvision->lock);
1170 return -EINVAL; 1183 return -EINVAL;
1171 } 1184 }
1172 1185
1173 for (i = 0; i < USBVISION_NUMFRAMES; i++) { 1186 for (i = 0; i < usbvision->num_frames; i++) {
1174 if (((usbvision->max_frame_size*i) >> PAGE_SHIFT) == vma->vm_pgoff) 1187 if (((PAGE_ALIGN(usbvision->max_frame_size)*i) >> PAGE_SHIFT) == vma->vm_pgoff)
1175 break; 1188 break;
1176 } 1189 }
1177 if (i == USBVISION_NUMFRAMES) { 1190 if (i == usbvision->num_frames) {
1178 PDEBUG(DBG_MMAP, "mmap: user supplied mapping address is out of range"); 1191 PDEBUG(DBG_MMAP, "mmap: user supplied mapping address is out of range");
1179 up(&usbvision->lock); 1192 up(&usbvision->lock);
1180 return -EINVAL; 1193 return -EINVAL;