Commit 3084920b authored by Thierry MERLE's avatar Thierry MERLE Committed by Mauro Carvalho Chehab

V4L/DVB (4929): Read() implementation + format set/get simplifications

- implement read() entry point that works with linux list.h
- rework of VIDIOC_ENUM_FMT/VIDIOC_S_FMT/VIDIOC_G_FMT
- VIDIOC_STREAMON : allows streaming whereas there is no queued buffer
(xdtv does VIDIOC_STREAMON before VIDIOC_QBUFs)
Signed-off-by: default avatarThierry MERLE <thierry.merle@free.fr>
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab@infradead.org>
parent 5f7fb877
...@@ -409,7 +409,8 @@ ...@@ -409,7 +409,8 @@
static int usbvision_nr = 0; // sequential number of usbvision device static int usbvision_nr = 0; // sequential number of usbvision device
static unsigned long usbvision_timestamp = 0; // timestamp in jiffies of a hundred frame
static unsigned long usbvision_counter = 0; // frame counter
static const int max_imgwidth = MAX_FRAME_WIDTH; static const int max_imgwidth = MAX_FRAME_WIDTH;
static const int max_imgheight = MAX_FRAME_HEIGHT; static const int max_imgheight = MAX_FRAME_HEIGHT;
...@@ -4278,6 +4279,7 @@ static int usbvision_v4l2_do_ioctl(struct inode *inode, struct file *file, ...@@ -4278,6 +4279,7 @@ static int usbvision_v4l2_do_ioctl(struct inode *inode, struct file *file,
list_add_tail(&usbvision->frame[vb->index].frame, &usbvision->inqueue); list_add_tail(&usbvision->frame[vb->index].frame, &usbvision->inqueue);
spin_unlock_irqrestore(&usbvision->queue_lock, lock_flags); spin_unlock_irqrestore(&usbvision->queue_lock, lock_flags);
PDEBUG(DBG_IOCTL, "VIDIOC_QBUF frame #%d",vb->index);
return 0; return 0;
} }
case VIDIOC_DQBUF: case VIDIOC_DQBUF:
...@@ -4312,18 +4314,31 @@ static int usbvision_v4l2_do_ioctl(struct inode *inode, struct file *file, ...@@ -4312,18 +4314,31 @@ static int usbvision_v4l2_do_ioctl(struct inode *inode, struct file *file,
vb->flags = V4L2_BUF_FLAG_MAPPED | V4L2_BUF_FLAG_QUEUED | V4L2_BUF_FLAG_DONE; vb->flags = V4L2_BUF_FLAG_MAPPED | V4L2_BUF_FLAG_QUEUED | V4L2_BUF_FLAG_DONE;
vb->index = f->index; vb->index = f->index;
vb->sequence = f->sequence; vb->sequence = f->sequence;
vb->timestamp = f->timestamp;
vb->field = V4L2_FIELD_NONE;
vb->bytesused = f->scanlength;
if(debug & DBG_IOCTL) { // do not spend computing time for debug stuff if not needed !
if(usbvision_counter == 100) {
PDEBUG(DBG_IOCTL, "VIDIOC_DQBUF delta=%d",(unsigned)(jiffies-usbvision_timestamp));
usbvision_counter = 0;
usbvision_timestamp = jiffies;
}
else {
usbvision_counter++;
}
PDEBUG(DBG_IOCTL, "VIDIOC_DQBUF frame #%d",vb->index);
}
return 0; return 0;
} }
case VIDIOC_STREAMON: case VIDIOC_STREAMON:
{ {
int b=V4L2_BUF_TYPE_VIDEO_CAPTURE; int b=V4L2_BUF_TYPE_VIDEO_CAPTURE;
if (list_empty(&usbvision->inqueue))
return -EINVAL;
usbvision->streaming = Stream_On; usbvision->streaming = Stream_On;
if(debug & DBG_IOCTL) usbvision_timestamp = jiffies;
call_i2c_clients(usbvision,VIDIOC_STREAMON , &b); call_i2c_clients(usbvision,VIDIOC_STREAMON , &b);
PDEBUG(DBG_IOCTL, "VIDIOC_STREAMON"); PDEBUG(DBG_IOCTL, "VIDIOC_STREAMON");
...@@ -4412,134 +4427,132 @@ static int usbvision_v4l2_do_ioctl(struct inode *inode, struct file *file, ...@@ -4412,134 +4427,132 @@ static int usbvision_v4l2_do_ioctl(struct inode *inode, struct file *file,
{ {
struct v4l2_fmtdesc *vfd = arg; struct v4l2_fmtdesc *vfd = arg;
if ( (dga == 0) &&
(vfd->type == V4L2_BUF_TYPE_VIDEO_OVERLAY) &&
(usbvision->palette.format != V4L2_PIX_FMT_YVU420) &&
(usbvision->palette.format != V4L2_PIX_FMT_YUV422P) ) {
return -EINVAL;
}
if(vfd->index>=USBVISION_SUPPORTED_PALETTES-1) { if(vfd->index>=USBVISION_SUPPORTED_PALETTES-1) {
return -EINVAL; return -EINVAL;
} }
vfd->flags = 0; vfd->flags = 0;
vfd->type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
strcpy(vfd->description,usbvision_v4l2_format[vfd->index].desc); strcpy(vfd->description,usbvision_v4l2_format[vfd->index].desc);
vfd->pixelformat = usbvision_v4l2_format[vfd->index].format; vfd->pixelformat = usbvision_v4l2_format[vfd->index].format;
memset(vfd->reserved, 0, sizeof(vfd->reserved));
return 0; return 0;
} }
case VIDIOC_G_FMT: case VIDIOC_G_FMT:
{ {
struct v4l2_format *vf = arg; struct v4l2_format *vf = arg;
if ( (dga == 0) && switch (vf->type) {
(vf->type == V4L2_BUF_TYPE_VIDEO_OVERLAY) && case V4L2_BUF_TYPE_VIDEO_CAPTURE:
(usbvision->palette.format != V4L2_PIX_FMT_YVU420) && {
(usbvision->palette.format != V4L2_PIX_FMT_YUV422P) ) { vf->fmt.pix.width = usbvision->curwidth;
return -EINVAL; vf->fmt.pix.height = usbvision->curheight;
vf->fmt.pix.pixelformat = usbvision->palette.format;
vf->fmt.pix.bytesperline = usbvision->curwidth*usbvision->palette.bytes_per_pixel;
vf->fmt.pix.sizeimage = vf->fmt.pix.bytesperline*usbvision->curheight;
vf->fmt.pix.colorspace = V4L2_COLORSPACE_SMPTE170M;
vf->fmt.pix.field = V4L2_FIELD_NONE; /* Always progressive image */
}
return 0;
default:
PDEBUG(DBG_IOCTL, "VIDIOC_G_FMT invalid type %d",vf->type);
return -EINVAL;
} }
down(&usbvision->lock); PDEBUG(DBG_IOCTL, "VIDIOC_G_FMT w=%d, h=%d",vf->fmt.win.w.width, vf->fmt.win.w.height);
*vf = usbvision->vid_win;
up(&usbvision->lock);
PDEBUG(DBG_IOCTL, "VIDIOC_G_FMT x=%d, y=%d, w=%d, h=%d, chroma=%x, clips=%d",
vf->fmt.win.w.left, vf->fmt.win.w.top, vf->fmt.win.w.width, vf->fmt.win.w.height, vf->fmt.win.chromakey, vf->fmt.win.clipcount);
return 0; return 0;
} }
case VIDIOC_TRY_FMT:
case VIDIOC_S_FMT: case VIDIOC_S_FMT:
{ {
struct v4l2_format *vf = arg; struct v4l2_format *vf = arg;
struct v4l2_clip *vc=NULL; struct v4l2_clip *vc=NULL;
int on,formatIdx; int on,formatIdx;
if ( (dga == 0) && switch(vf->type) {
(vf->type == V4L2_BUF_TYPE_VIDEO_OVERLAY) && case V4L2_BUF_TYPE_VIDEO_OVERLAY:
(usbvision->palette.format != V4L2_PIX_FMT_YVU420) && {
(usbvision->palette.format != V4L2_PIX_FMT_YUV422P) ) { if (vf->fmt.win.clipcount>256) {
return -EINVAL; return -EDOM; /* Too many clips! */
} }
if(vf->type == V4L2_BUF_TYPE_VIDEO_OVERLAY) { // Do every clips.
if (vf->fmt.win.clipcount>256) { vc = vmalloc(sizeof(struct v4l2_clip)*(vf->fmt.win.clipcount+4));
return -EDOM; /* Too many clips! */ if (vc == NULL) {
} return -ENOMEM;
// Do every clips. }
vc = vmalloc(sizeof(struct v4l2_clip)*(vf->fmt.win.clipcount+4)); if (vf->fmt.win.clipcount && copy_from_user(vc,vf->fmt.win.clips,sizeof(struct v4l2_clip)*vf->fmt.win.clipcount)) {
if (vc == NULL) { return -EFAULT;
return -ENOMEM; }
} on = usbvision->overlay; // Save overlay state
if (vf->fmt.win.clipcount && copy_from_user(vc,vf->fmt.win.clips,sizeof(struct v4l2_clip)*vf->fmt.win.clipcount)) { if (on) {
return -EFAULT; usbvision_cap(usbvision, 0);
} }
on = usbvision->overlay; // Save overlay state
if (on) {
usbvision_cap(usbvision, 0);
}
// strange, it seems xawtv sometimes calls us with 0 // strange, it seems xawtv sometimes calls us with 0
// width and/or height. Ignore these values // width and/or height. Ignore these values
if (vf->fmt.win.w.left == 0) { if (vf->fmt.win.w.left == 0) {
vf->fmt.win.w.left = usbvision->vid_win.fmt.win.w.left; vf->fmt.win.w.left = usbvision->vid_win.fmt.win.w.left;
} }
if (vf->fmt.win.w.top == 0) { if (vf->fmt.win.w.top == 0) {
vf->fmt.win.w.top = usbvision->vid_win.fmt.win.w.top; vf->fmt.win.w.top = usbvision->vid_win.fmt.win.w.top;
} }
// by now we are committed to the new data... // by now we are committed to the new data...
down(&usbvision->lock); down(&usbvision->lock);
RESTRICT_TO_RANGE(vf->fmt.win.w.width, MIN_FRAME_WIDTH, MAX_FRAME_WIDTH); RESTRICT_TO_RANGE(vf->fmt.win.w.width, MIN_FRAME_WIDTH, MAX_FRAME_WIDTH);
RESTRICT_TO_RANGE(vf->fmt.win.w.height, MIN_FRAME_HEIGHT, MAX_FRAME_HEIGHT); RESTRICT_TO_RANGE(vf->fmt.win.w.height, MIN_FRAME_HEIGHT, MAX_FRAME_HEIGHT);
usbvision->vid_win = *vf; usbvision->vid_win = *vf;
usbvision->overlay_frame.width = vf->fmt.win.w.width; usbvision->overlay_frame.width = vf->fmt.win.w.width;
usbvision->overlay_frame.height = vf->fmt.win.w.height; usbvision->overlay_frame.height = vf->fmt.win.w.height;
usbvision_set_output(usbvision, vf->fmt.win.w.width, vf->fmt.win.w.height); usbvision_set_output(usbvision, vf->fmt.win.w.width, vf->fmt.win.w.height);
up(&usbvision->lock); up(&usbvision->lock);
// Impose display clips // Impose display clips
if (vf->fmt.win.w.left+vf->fmt.win.w.width > (unsigned int)usbvision->vid_buf.fmt.width) { if (vf->fmt.win.w.left+vf->fmt.win.w.width > (unsigned int)usbvision->vid_buf.fmt.width) {
usbvision_new_clip(vf, vc, usbvision->vid_buf.fmt.width-vf->fmt.win.w.left, 0, vf->fmt.win.w.width-1, vf->fmt.win.w.height-1); usbvision_new_clip(vf, vc, usbvision->vid_buf.fmt.width-vf->fmt.win.w.left, 0, vf->fmt.win.w.width-1, vf->fmt.win.w.height-1);
} }
if (vf->fmt.win.w.top+vf->fmt.win.w.height > (unsigned int)usbvision->vid_buf.fmt.height) { if (vf->fmt.win.w.top+vf->fmt.win.w.height > (unsigned int)usbvision->vid_buf.fmt.height) {
usbvision_new_clip(vf, vc, 0, usbvision->vid_buf.fmt.height-vf->fmt.win.w.top, vf->fmt.win.w.width-1, vf->fmt.win.w.height-1); usbvision_new_clip(vf, vc, 0, usbvision->vid_buf.fmt.height-vf->fmt.win.w.top, vf->fmt.win.w.width-1, vf->fmt.win.w.height-1);
} }
// built the requested clipping zones // built the requested clipping zones
usbvision_built_overlay(usbvision, vf->fmt.win.clipcount, vc); usbvision_built_overlay(usbvision, vf->fmt.win.clipcount, vc);
vfree(vc); vfree(vc);
// restore overlay state // restore overlay state
if (on) { if (on) {
usbvision_cap(usbvision, 1); usbvision_cap(usbvision, 1);
}
usbvision->vid_win_valid = 1;
PDEBUG(DBG_IOCTL, "VIDIOC_S_FMT overlay x=%d, y=%d, w=%d, h=%d, chroma=%x, clips=%d",
vf->fmt.win.w.left, vf->fmt.win.w.top, vf->fmt.win.w.width, vf->fmt.win.w.height, vf->fmt.win.chromakey, vf->fmt.win.clipcount);
return 0;
} }
usbvision->vid_win_valid = 1; case V4L2_BUF_TYPE_VIDEO_CAPTURE:
PDEBUG(DBG_IOCTL, "VIDIOC_S_FMT overlay x=%d, y=%d, w=%d, h=%d, chroma=%x, clips=%d", {
vf->fmt.win.w.left, vf->fmt.win.w.top, vf->fmt.win.w.width, vf->fmt.win.w.height, vf->fmt.win.chromakey, vf->fmt.win.clipcount); /* Find requested format in available ones */
} for(formatIdx=0;formatIdx<USBVISION_SUPPORTED_PALETTES;formatIdx++) {
else { if(vf->fmt.pix.pixelformat == usbvision_v4l2_format[formatIdx].format) {
/* Find requested format in available ones */ usbvision->palette = usbvision_v4l2_format[formatIdx];
for(formatIdx=0;formatIdx<USBVISION_SUPPORTED_PALETTES;formatIdx++) { break;
if(vf->fmt.pix.pixelformat == usbvision_v4l2_format[formatIdx].format) { }
usbvision->palette = usbvision_v4l2_format[formatIdx]; }
break; /* robustness */
if(formatIdx == USBVISION_SUPPORTED_PALETTES) {
return -EINVAL;
} }
RESTRICT_TO_RANGE(vf->fmt.pix.width, MIN_FRAME_WIDTH, MAX_FRAME_WIDTH);
RESTRICT_TO_RANGE(vf->fmt.pix.height, MIN_FRAME_HEIGHT, MAX_FRAME_HEIGHT);
// by now we are committed to the new data...
down(&usbvision->lock);
usbvision_set_output(usbvision, vf->fmt.pix.width, vf->fmt.pix.height);
up(&usbvision->lock);
PDEBUG(DBG_IOCTL, "VIDIOC_S_FMT grabdisplay w=%d, h=%d, format=%s",
vf->fmt.pix.width, vf->fmt.pix.height,usbvision->palette.desc);
return 0;
} }
/* robustness */ default:
if(formatIdx == USBVISION_SUPPORTED_PALETTES) {
return -EINVAL; return -EINVAL;
}
usbvision->vid_win.fmt.pix.pixelformat = vf->fmt.pix.pixelformat;
usbvision->vid_win.fmt.pix.width = vf->fmt.pix.width; //Image width in pixels.
usbvision->vid_win.fmt.pix.height = vf->fmt.pix.height; // Image height in pixels.
// by now we are committed to the new data...
down(&usbvision->lock);
RESTRICT_TO_RANGE(vf->fmt.pix.width, MIN_FRAME_WIDTH, MAX_FRAME_WIDTH);
RESTRICT_TO_RANGE(vf->fmt.pix.height, MIN_FRAME_HEIGHT, MAX_FRAME_HEIGHT);
usbvision->vid_win = *vf;
usbvision_set_output(usbvision, vf->fmt.pix.width, vf->fmt.pix.height);
up(&usbvision->lock);
usbvision->vid_win_valid = 1;
PDEBUG(DBG_IOCTL, "VIDIOC_S_FMT grabdisplay w=%d, h=%d, ",
vf->fmt.pix.width, vf->fmt.pix.height);
} }
return 0;
} }
case VIDIOC_OVERLAY: case VIDIOC_OVERLAY:
{ {
...@@ -4586,99 +4599,97 @@ static ssize_t usbvision_v4l2_read(struct file *file, char *buf, ...@@ -4586,99 +4599,97 @@ static ssize_t usbvision_v4l2_read(struct file *file, char *buf,
struct video_device *dev = video_devdata(file); struct video_device *dev = video_devdata(file);
struct usb_usbvision *usbvision = (struct usb_usbvision *) video_get_drvdata(dev); struct usb_usbvision *usbvision = (struct usb_usbvision *) video_get_drvdata(dev);
int noblock = file->f_flags & O_NONBLOCK; int noblock = file->f_flags & O_NONBLOCK;
unsigned long lock_flags;
int frmx = -1; int frmx = -1;
int rc = 0; int ret,i;
struct usbvision_frame *frame; struct usbvision_frame *frame;
return -EINVAL;
PDEBUG(DBG_IO, "%s: %ld bytes, noblock=%d", __FUNCTION__, (unsigned long)count, noblock); PDEBUG(DBG_IO, "%s: %ld bytes, noblock=%d", __FUNCTION__, (unsigned long)count, noblock);
if (!USBVISION_IS_OPERATIONAL(usbvision) || (buf == NULL)) if (!USBVISION_IS_OPERATIONAL(usbvision) || (buf == NULL))
return -EFAULT; return -EFAULT;
down(&usbvision->lock); /* no stream is running, make it running ! */
//code for testing compression usbvision->streaming = Stream_On;
if (usbvision->isocMode == ISOC_MODE_COMPRESS) { call_i2c_clients(usbvision,VIDIOC_STREAMON , NULL);
usbvision->frame[0].v4l2_format = usbvision_v4l2_format[0]; //V4L2_PIX_FMT_GREY;
usbvision->frame[1].v4l2_format = usbvision_v4l2_format[0]; // V4L2_PIX_FMT_GREY;
}
// See if a frame is completed, then use it. /* First, enqueue as many frames as possible (like a user of VIDIOC_QBUF would do) */
if (usbvision->frame[0].grabstate >= FrameState_Done) // _Done or _Error for(i=0;i<USBVISION_NUMFRAMES;i++) {
frmx = 0; frame = &usbvision->frame[i];
else if (usbvision->frame[1].grabstate >= FrameState_Done)// _Done or _Error if(frame->grabstate == FrameState_Unused) {
frmx = 1; /* Mark it as ready and enqueue frame */
frame->grabstate = FrameState_Ready;
frame->scanstate = ScanState_Scanning;
frame->scanlength = 0; /* Accumulated in usbvision_parse_data() */
if (noblock && (frmx == -1)) { /* set v4l2_format index */
count = -EAGAIN; frame->v4l2_format = usbvision->palette;
goto usbvision_v4l2_read_done;
}
// If no FRAME_DONE, look for a FRAME_GRABBING state. spin_lock_irqsave(&usbvision->queue_lock, lock_flags);
// See if a frame is in process (grabbing), then use it. list_add_tail(&frame->frame, &usbvision->inqueue);
if (frmx == -1) { spin_unlock_irqrestore(&usbvision->queue_lock, lock_flags);
if (usbvision->frame[0].grabstate == FrameState_Grabbing) }
frmx = 0;
else if (usbvision->frame[1].grabstate == FrameState_Grabbing)
frmx = 1;
} }
// If no frame is active, start one. /* Then try to steal a frame (like a VIDIOC_DQBUF would do) */
if (frmx == -1) if (list_empty(&(usbvision->outqueue))) {
// FIXME: enqueue all inqueue... if(noblock)
/* usbvision_new_frame(usbvision, frmx = 0); */ return -EAGAIN;
frame = &usbvision->frame[frmx];
restart: ret = wait_event_interruptible
if (!USBVISION_IS_OPERATIONAL(usbvision)) { (usbvision->wait_frame,
count = -EIO; !list_empty(&(usbvision->outqueue)));
goto usbvision_v4l2_read_done; if (ret)
return ret;
} }
PDEBUG(DBG_IO, "Waiting frame grabbing");
rc = wait_event_interruptible(usbvision->wait_frame, (frame->grabstate == FrameState_Done) || spin_lock_irqsave(&usbvision->queue_lock, lock_flags);
(frame->grabstate == FrameState_Error)); frame = list_entry(usbvision->outqueue.next,
if (rc) { struct usbvision_frame, frame);
goto usbvision_v4l2_read_done; list_del(usbvision->outqueue.next);
spin_unlock_irqrestore(&usbvision->queue_lock, lock_flags);
if(debug & DBG_IOCTL) { // do not spend computing time for debug stuff if not needed !
if(usbvision_counter == 100) {
PDEBUG(DBG_IOCTL, "VIDIOC_DQBUF delta=%d",(unsigned)(jiffies-usbvision_timestamp));
usbvision_counter = 0;
usbvision_timestamp = jiffies;
}
else {
usbvision_counter++;
}
} }
/* An error returns an empty frame */
if (frame->grabstate == FrameState_Error) { if (frame->grabstate == FrameState_Error) {
frame->bytes_read = 0; frame->bytes_read = 0;
// FIXME: enqueue all inqueue... return 0;
/* if (usbvision_new_frame(usbvision, frmx)) { */
/* err("%s: usbvision_new_frame() failed", __FUNCTION__); */
/* } */
goto restart;
} }
PDEBUG(DBG_IO, "%s: frmx=%d, bytes_read=%ld, scanlength=%ld", __FUNCTION__, PDEBUG(DBG_IO, "%s: frmx=%d, bytes_read=%ld, scanlength=%ld", __FUNCTION__,
frmx, frame->bytes_read, frame->scanlength); frame->index, frame->bytes_read, frame->scanlength);
/* copy bytes to user space; we allow for partials reads */ /* copy bytes to user space; we allow for partials reads */
if ((count + frame->bytes_read) > (unsigned long)frame->scanlength) if ((count + frame->bytes_read) > (unsigned long)frame->scanlength)
count = frame->scanlength - frame->bytes_read; count = frame->scanlength - frame->bytes_read;
if (copy_to_user(buf, frame->data + frame->bytes_read, count)) { if (copy_to_user(buf, frame->data + frame->bytes_read, count)) {
count = -EFAULT; return -EFAULT;
goto usbvision_v4l2_read_done;
} }
frame->bytes_read += count; frame->bytes_read += count;
PDEBUG(DBG_IO, "%s: {copy} count used=%ld, new bytes_read=%ld", __FUNCTION__, PDEBUG(DBG_IO, "%s: {copy} count used=%ld, new bytes_read=%ld", __FUNCTION__,
(unsigned long)count, frame->bytes_read); (unsigned long)count, frame->bytes_read);
if (frame->bytes_read >= frame->scanlength) {// All data has been read // For now, forget the frame if it has not been read in one shot.
/* if (frame->bytes_read >= frame->scanlength) {// All data has been read */
frame->bytes_read = 0; frame->bytes_read = 0;
/* Mark it as available to be used again. */ /* Mark it as available to be used again. */
usbvision->frame[frmx].grabstate = FrameState_Unused; usbvision->frame[frmx].grabstate = FrameState_Unused;
// FIXME enqueue another frame /* } */
/* if (usbvision_new_frame(usbvision, frmx ? 0 : 1)) */
/* err("%s: usbvision_new_frame() failed", __FUNCTION__); */
}
usbvision_v4l2_read_done:
up(&usbvision->lock);
return count; return count;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment