diff options
author | Liu Ying <Ying.Liu@freescale.com> | 2013-04-23 12:24:17 +0800 |
---|---|---|
committer | Liu Ying <Ying.Liu@freescale.com> | 2013-05-02 17:03:36 +0800 |
commit | 1e00529f451b20de9633eb1ea1a9b9f3121dbdd3 (patch) | |
tree | 5015a0988e3b99a14b3aeaca204bb67660e9b406 | |
parent | 73c759092a5fc07fff18120bb8103b2e3b3949cf (diff) |
ENGR00259949 mxc vout:fix screen tearing issue in ic bypass case
In ic bypass case, we put video buffers at a framebuffer display
channel directly. The display channel works at triple buffer
mode. To make sure a video buffer(buf N) has been shown on display
device, we at least need to wait for the second video buffer(buf N+2)
after the current buffer(buf N) is put on the display channel. Then,
the current buffer(buf N) can be added to the dequeue list, otherwise,
the user may get the buffer too early so that the buffer being shown
can be overwritten - screen tearing issue happens.
Signed-off-by: Liu Ying <Ying.Liu@freescale.com>
-rw-r--r-- | drivers/media/video/mxc/output/mxc_vout.c | 33 |
1 files changed, 19 insertions, 14 deletions
diff --git a/drivers/media/video/mxc/output/mxc_vout.c b/drivers/media/video/mxc/output/mxc_vout.c index 719936bd516b..677a7b011989 100644 --- a/drivers/media/video/mxc/output/mxc_vout.c +++ b/drivers/media/video/mxc/output/mxc_vout.c @@ -129,7 +129,8 @@ struct mxc_vout_output { dma_addr_t disp_bufs[FB_BUFS]; - struct videobuf_buffer *pre_vb; + struct videobuf_buffer *pre1_vb; + struct videobuf_buffer *pre2_vb; }; struct mxc_vout_dev { @@ -727,21 +728,24 @@ vdi_frame_rate_double: list_del(&vb->queue); /* - * previous videobuf finish show, set VIDEOBUF_DONE state here - * to avoid tearing issue in pp bypass case, which make sure - * showing buffer will not be dequeue to write new data. It also - * bring side-effect that the last buffer can not be dequeue - * correctly, app need take care about it. + * The videobuf before the last one has been shown. Set + * VIDEOBUF_DONE state here to avoid tearing issue in ic bypass + * case, which makes sure a buffer being shown will not be + * dequeued to be overwritten. It also brings side-effect that + * the last 2 buffers can not be dequeued correctly, apps need + * to take care of it. */ - if (vout->pre_vb) { - vout->pre_vb->state = VIDEOBUF_DONE; - wake_up_interruptible(&vout->pre_vb->done); + if (vout->pre2_vb) { + vout->pre2_vb->state = VIDEOBUF_DONE; + wake_up_interruptible(&vout->pre2_vb->done); } - if (vout->linear_bypass_pp) - vout->pre_vb = vb; - else { - vout->pre_vb = NULL; + if (vout->linear_bypass_pp) { + vout->pre2_vb = vout->pre1_vb; + vout->pre1_vb = vb; + } else { + vout->pre1_vb = NULL; + vout->pre2_vb = NULL; vb->state = VIDEOBUF_DONE; wake_up_interruptible(&vb->done); } @@ -1897,7 +1901,8 @@ static int mxc_vidioc_streamon(struct file *file, void *fh, vout->start_jiffies = jiffies; - vout->pre_vb = NULL; + vout->pre1_vb = NULL; + vout->pre2_vb = NULL; ret = videobuf_streamon(q); done: |