summaryrefslogtreecommitdiff
path: root/drivers/media
diff options
context:
space:
mode:
authorJason Chen <b02280@freescale.com>2011-11-15 12:54:26 +0800
committerJason Liu <r64343@freescale.com>2012-01-09 21:06:34 +0800
commit3235f430dfc789c552c69755919d10e43828ef57 (patch)
tree5fbc4987fb355bce25e4d4b03db7b3d44f12f672 /drivers/media
parent91ec57952d686e602e427da33fcc299aca0b43d8 (diff)
ENGR00162064 mxc v4l2 output: check ipu task in S_CROP ioctl
In mxc_vout, ioctl S_CROP should call before S_FMT, for fist time S_CROP, will not check ipu task, it actually will check in S_FMT, after S_FMT, S_CROP should check ipu task too. Signed-off-by: Jason Chen <b02280@freescale.com>
Diffstat (limited to 'drivers/media')
-rw-r--r--drivers/media/video/mxc/output/mxc_vout.c94
1 files changed, 60 insertions, 34 deletions
diff --git a/drivers/media/video/mxc/output/mxc_vout.c b/drivers/media/video/mxc/output/mxc_vout.c
index fb82a6816e83..fe4a2bd617ea 100644
--- a/drivers/media/video/mxc/output/mxc_vout.c
+++ b/drivers/media/video/mxc/output/mxc_vout.c
@@ -60,6 +60,7 @@ struct mxc_vout_output {
bool disp_support_windows;
bool disp_support_csc;
+ bool fmt_init;
struct ipu_task task;
bool timer_stop;
@@ -401,7 +402,7 @@ static void disp_work_func(struct work_struct *work)
struct videobuf_queue *q = &vout->vbq;
struct videobuf_buffer *vb, *vb_next = NULL;
unsigned long flags = 0;
- int ret;
+ int ret = 0;
spin_lock_irqsave(q->irqlock, flags);
@@ -453,8 +454,9 @@ static void disp_work_func(struct work_struct *work)
goto err;
}
- if (show_buf(vout, vout->frame_count % FB_BUFS) < 0)
- goto err;
+ ret = show_buf(vout, vout->frame_count % FB_BUFS);
+ if (ret < 0)
+ v4l2_warn(vout->vfd->v4l2_dev, "show buf with ret %d\n", ret);
spin_lock_irqsave(q->irqlock, flags);
@@ -491,7 +493,7 @@ static void disp_work_func(struct work_struct *work)
return;
err:
- v4l2_err(vout->vfd->v4l2_dev, "display work fail\n");
+ v4l2_err(vout->vfd->v4l2_dev, "display work fail ret = %d\n", ret);
vout->timer_stop = true;
vb->state = VIDEOBUF_ERROR;
return;
@@ -681,6 +683,7 @@ static int mxc_vout_open(struct file *file)
INIT_LIST_HEAD(&vout->queue_list);
INIT_LIST_HEAD(&vout->active_list);
+ vout->fmt_init = false;
vout->frame_count = 0;
vout->win_pos.x = 0;
@@ -777,6 +780,36 @@ again:
return ret;
}
+static int mxc_vout_try_task(struct mxc_vout_output *vout)
+{
+ int ret = 0;
+
+ vout->task.input.crop.w -= vout->task.input.crop.w%8;
+ vout->task.input.crop.h -= vout->task.input.crop.h%8;
+
+ /* assume task.output already set by S_CROP */
+ if (is_pp_bypass(vout)) {
+ v4l2_info(vout->vfd->v4l2_dev, "Bypass IC.\n");
+ vout->task.output.format = vout->task.input.format;
+ } else {
+ /* if need CSC, choose IPU-DP or IPU_IC do it */
+ if (vout->disp_support_csc) {
+ if (colorspaceofpixel(vout->task.input.format) == YUV_CS)
+ vout->task.output.format = IPU_PIX_FMT_UYVY;
+ else
+ vout->task.output.format = IPU_PIX_FMT_RGB565;
+ } else {
+ if (colorspaceofpixel(vout->disp_fmt) == YUV_CS)
+ vout->task.output.format = IPU_PIX_FMT_UYVY;
+ else
+ vout->task.output.format = IPU_PIX_FMT_RGB565;
+ }
+ ret = ipu_try_task(&vout->task);
+ }
+
+ return ret;
+}
+
static int mxc_vout_try_format(struct mxc_vout_output *vout, struct v4l2_format *f)
{
int ret = 0;
@@ -816,41 +849,23 @@ static int mxc_vout_try_format(struct mxc_vout_output *vout, struct v4l2_format
rect = (struct v4l2_rect *)f->fmt.pix.priv;
vout->task.input.crop.pos.x = rect->left;
vout->task.input.crop.pos.y = rect->top;
- vout->task.input.crop.w = rect->width - rect->width%8;
- vout->task.input.crop.h = rect->height - rect->height%8;
+ vout->task.input.crop.w = rect->width;
+ vout->task.input.crop.h = rect->height;
} else {
vout->task.input.crop.pos.x = 0;
vout->task.input.crop.pos.y = 0;
- vout->task.input.crop.w = f->fmt.pix.width - f->fmt.pix.width%8;
- vout->task.input.crop.h = f->fmt.pix.height - f->fmt.pix.height%8;
+ vout->task.input.crop.w = f->fmt.pix.width;
+ vout->task.input.crop.h = f->fmt.pix.height;
}
- /* assume task.output already set by S_CROP */
- if (is_pp_bypass(vout)) {
- v4l2_info(vout->vfd->v4l2_dev, "Bypass IC.\n");
- vout->task.output.format = vout->task.input.format;
- } else {
- /* if need CSC, choose IPU-DP or IPU_IC do it */
- if (vout->disp_support_csc) {
- if (colorspaceofpixel(vout->task.input.format) == YUV_CS)
- vout->task.output.format = IPU_PIX_FMT_UYVY;
- else
- vout->task.output.format = IPU_PIX_FMT_RGB565;
+ ret = mxc_vout_try_task(vout);
+ if (!ret) {
+ if (rect) {
+ rect->width = vout->task.input.crop.w;
+ rect->height = vout->task.input.crop.h;
} else {
- if (colorspaceofpixel(vout->disp_fmt) == YUV_CS)
- vout->task.output.format = IPU_PIX_FMT_UYVY;
- else
- vout->task.output.format = IPU_PIX_FMT_RGB565;
- }
- ret = ipu_try_task(&vout->task);
- if (!ret) {
- if (rect) {
- rect->width = vout->task.input.crop.w;
- rect->height = vout->task.input.crop.h;
- } else {
- f->fmt.pix.width = vout->task.input.crop.w;
- f->fmt.pix.height = vout->task.input.crop.h;
- }
+ f->fmt.pix.width = vout->task.input.crop.w;
+ f->fmt.pix.height = vout->task.input.crop.h;
}
}
@@ -868,6 +883,8 @@ static int mxc_vidioc_s_fmt_vid_out(struct file *file, void *fh,
mutex_lock(&vout->task_lock);
ret = mxc_vout_try_format(vout, f);
+ if (ret >= 0)
+ vout->fmt_init = true;
mutex_unlock(&vout->task_lock);
return ret;
@@ -913,6 +930,7 @@ static int mxc_vidioc_s_crop(struct file *file, void *fh, struct v4l2_crop *crop
{
struct mxc_vout_output *vout = fh;
struct v4l2_rect *b = &vout->crop_bounds;
+ int ret = 0;
if (crop->type != V4L2_BUF_TYPE_VIDEO_OUTPUT)
return -EINVAL;
@@ -965,9 +983,17 @@ static int mxc_vidioc_s_crop(struct file *file, void *fh, struct v4l2_crop *crop
vout->task.output.crop.w = crop->c.width;
vout->task.output.crop.h = crop->c.height;
+ /*
+ * must S_CROP before S_FMT, for fist time S_CROP, will not check
+ * ipu task, it will check in S_FMT, after S_FMT, S_CROP should
+ * check ipu task too.
+ */
+ if (vout->fmt_init)
+ ret = mxc_vout_try_task(vout);
+
mutex_unlock(&vout->task_lock);
- return 0;
+ return ret;
}
static int mxc_vidioc_queryctrl(struct file *file, void *fh,