2011-06-17 07:01:31 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2010 Texas Instruments Incorporated - http://www.ti.com/
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License as
|
|
|
|
* published by the Free Software Foundation version 2.
|
|
|
|
*
|
|
|
|
* This program is distributed WITHOUT ANY WARRANTY of any
|
|
|
|
* kind, whether express or implied; without even the implied warranty
|
|
|
|
* of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*/
|
|
|
|
#include <linux/kernel.h>
|
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/errno.h>
|
|
|
|
#include <linux/interrupt.h>
|
|
|
|
#include <linux/string.h>
|
|
|
|
#include <linux/wait.h>
|
|
|
|
#include <linux/time.h>
|
|
|
|
#include <linux/platform_device.h>
|
|
|
|
#include <linux/irq.h>
|
|
|
|
#include <linux/mm.h>
|
|
|
|
#include <linux/mutex.h>
|
|
|
|
#include <linux/videodev2.h>
|
|
|
|
#include <linux/slab.h>
|
|
|
|
|
|
|
|
#include <asm/pgtable.h>
|
2018-04-05 16:31:39 +00:00
|
|
|
|
|
|
|
#ifdef CONFIG_ARCH_DAVINCI
|
2011-06-17 07:01:31 +00:00
|
|
|
#include <mach/cputype.h>
|
2018-04-05 16:31:39 +00:00
|
|
|
#endif
|
2011-06-17 07:01:31 +00:00
|
|
|
|
|
|
|
#include <media/v4l2-dev.h>
|
|
|
|
#include <media/v4l2-common.h>
|
|
|
|
#include <media/v4l2-ioctl.h>
|
|
|
|
#include <media/v4l2-device.h>
|
|
|
|
#include <media/davinci/vpbe_display.h>
|
|
|
|
#include <media/davinci/vpbe_types.h>
|
|
|
|
#include <media/davinci/vpbe.h>
|
|
|
|
#include <media/davinci/vpbe_venc.h>
|
|
|
|
#include <media/davinci/vpbe_osd.h>
|
|
|
|
#include "vpbe_venc_regs.h"
|
|
|
|
|
|
|
|
#define VPBE_DISPLAY_DRIVER "vpbe-v4l2"
|
|
|
|
|
|
|
|
static int debug;
|
|
|
|
|
|
|
|
#define VPBE_DEFAULT_NUM_BUFS 3
|
|
|
|
|
|
|
|
module_param(debug, int, 0644);
|
|
|
|
|
2012-10-22 12:27:13 +00:00
|
|
|
static int vpbe_set_osd_display_params(struct vpbe_display *disp_dev,
|
|
|
|
struct vpbe_layer *layer);
|
|
|
|
|
2011-06-17 07:01:31 +00:00
|
|
|
static int venc_is_second_field(struct vpbe_display *disp_dev)
|
|
|
|
{
|
|
|
|
struct vpbe_device *vpbe_dev = disp_dev->vpbe_dev;
|
media: vpbe_display: properly handle error case
if v4l2_subdev_call(..., VENC_GET_FLD,...) fails, it
currently returns a random value. Instead, return 1.
That's probably better than returning 0, as this is very
likely what happens in practice with the current code, as
as the probably of an unititialized 32 bits integer to
have an specific value (0, in this case), is 1/(2^32).
An alternative would be to return an error code, and
let the caller to hint, based on the past received
frame, but that sounds weird.
Signed-off-by: Mauro Carvalho Chehab <mchehab@s-opensource.com>
2018-04-06 12:12:51 +00:00
|
|
|
int ret, val;
|
2011-06-17 07:01:31 +00:00
|
|
|
|
|
|
|
ret = v4l2_subdev_call(vpbe_dev->venc,
|
|
|
|
core,
|
|
|
|
ioctl,
|
|
|
|
VENC_GET_FLD,
|
|
|
|
&val);
|
|
|
|
if (ret < 0) {
|
|
|
|
v4l2_err(&vpbe_dev->v4l2_dev,
|
|
|
|
"Error in getting Field ID 0\n");
|
media: vpbe_display: properly handle error case
if v4l2_subdev_call(..., VENC_GET_FLD,...) fails, it
currently returns a random value. Instead, return 1.
That's probably better than returning 0, as this is very
likely what happens in practice with the current code, as
as the probably of an unititialized 32 bits integer to
have an specific value (0, in this case), is 1/(2^32).
An alternative would be to return an error code, and
let the caller to hint, based on the past received
frame, but that sounds weird.
Signed-off-by: Mauro Carvalho Chehab <mchehab@s-opensource.com>
2018-04-06 12:12:51 +00:00
|
|
|
return 1;
|
2011-06-17 07:01:31 +00:00
|
|
|
}
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void vpbe_isr_even_field(struct vpbe_display *disp_obj,
|
|
|
|
struct vpbe_layer *layer)
|
|
|
|
{
|
|
|
|
if (layer->cur_frm == layer->next_frm)
|
|
|
|
return;
|
2015-05-26 14:20:27 +00:00
|
|
|
|
2015-11-03 10:16:37 +00:00
|
|
|
layer->cur_frm->vb.vb2_buf.timestamp = ktime_get_ns();
|
[media] media: videobuf2: Restructure vb2_buffer
Remove v4l2 stuff - v4l2_buf, v4l2_plane - from struct vb2_buffer.
Add new member variables - bytesused, length, offset, userptr, fd,
data_offset - to struct vb2_plane in order to cover all information
of v4l2_plane.
struct vb2_plane {
<snip>
unsigned int bytesused;
unsigned int length;
union {
unsigned int offset;
unsigned long userptr;
int fd;
} m;
unsigned int data_offset;
}
Replace v4l2_buf with new member variables - index, type, memory - which
are common fields for buffer management.
struct vb2_buffer {
<snip>
unsigned int index;
unsigned int type;
unsigned int memory;
unsigned int num_planes;
struct vb2_plane planes[VIDEO_MAX_PLANES];
<snip>
};
v4l2 specific fields - flags, field, timestamp, timecode,
sequence - are moved to vb2_v4l2_buffer in videobuf2-v4l2.c
struct vb2_v4l2_buffer {
struct vb2_buffer vb2_buf;
__u32 flags;
__u32 field;
struct timeval timestamp;
struct v4l2_timecode timecode;
__u32 sequence;
};
Signed-off-by: Junghak Sung <jh1009.sung@samsung.com>
Signed-off-by: Geunyoung Kim <nenggun.kim@samsung.com>
Acked-by: Seung-Woo Kim <sw0312.kim@samsung.com>
Acked-by: Inki Dae <inki.dae@samsung.com>
Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
2015-09-22 13:30:30 +00:00
|
|
|
vb2_buffer_done(&layer->cur_frm->vb.vb2_buf, VB2_BUF_STATE_DONE);
|
2011-06-17 07:01:31 +00:00
|
|
|
/* Make cur_frm pointing to next_frm */
|
|
|
|
layer->cur_frm = layer->next_frm;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void vpbe_isr_odd_field(struct vpbe_display *disp_obj,
|
|
|
|
struct vpbe_layer *layer)
|
|
|
|
{
|
|
|
|
struct osd_state *osd_device = disp_obj->osd_device;
|
|
|
|
unsigned long addr;
|
|
|
|
|
|
|
|
spin_lock(&disp_obj->dma_queue_lock);
|
|
|
|
if (list_empty(&layer->dma_queue) ||
|
|
|
|
(layer->cur_frm != layer->next_frm)) {
|
|
|
|
spin_unlock(&disp_obj->dma_queue_lock);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* one field is displayed configure
|
|
|
|
* the next frame if it is available
|
|
|
|
* otherwise hold on current frame
|
|
|
|
* Get next from the buffer queue
|
|
|
|
*/
|
2012-10-22 12:27:13 +00:00
|
|
|
layer->next_frm = list_entry(layer->dma_queue.next,
|
|
|
|
struct vpbe_disp_buffer, list);
|
2011-06-17 07:01:31 +00:00
|
|
|
/* Remove that from the buffer queue */
|
2012-10-22 12:27:13 +00:00
|
|
|
list_del(&layer->next_frm->list);
|
2011-06-17 07:01:31 +00:00
|
|
|
spin_unlock(&disp_obj->dma_queue_lock);
|
|
|
|
/* Mark state of the frame to active */
|
[media] media: videobuf2: Restructure vb2_buffer
Remove v4l2 stuff - v4l2_buf, v4l2_plane - from struct vb2_buffer.
Add new member variables - bytesused, length, offset, userptr, fd,
data_offset - to struct vb2_plane in order to cover all information
of v4l2_plane.
struct vb2_plane {
<snip>
unsigned int bytesused;
unsigned int length;
union {
unsigned int offset;
unsigned long userptr;
int fd;
} m;
unsigned int data_offset;
}
Replace v4l2_buf with new member variables - index, type, memory - which
are common fields for buffer management.
struct vb2_buffer {
<snip>
unsigned int index;
unsigned int type;
unsigned int memory;
unsigned int num_planes;
struct vb2_plane planes[VIDEO_MAX_PLANES];
<snip>
};
v4l2 specific fields - flags, field, timestamp, timecode,
sequence - are moved to vb2_v4l2_buffer in videobuf2-v4l2.c
struct vb2_v4l2_buffer {
struct vb2_buffer vb2_buf;
__u32 flags;
__u32 field;
struct timeval timestamp;
struct v4l2_timecode timecode;
__u32 sequence;
};
Signed-off-by: Junghak Sung <jh1009.sung@samsung.com>
Signed-off-by: Geunyoung Kim <nenggun.kim@samsung.com>
Acked-by: Seung-Woo Kim <sw0312.kim@samsung.com>
Acked-by: Inki Dae <inki.dae@samsung.com>
Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
2015-09-22 13:30:30 +00:00
|
|
|
layer->next_frm->vb.vb2_buf.state = VB2_BUF_STATE_ACTIVE;
|
|
|
|
addr = vb2_dma_contig_plane_dma_addr(&layer->next_frm->vb.vb2_buf, 0);
|
2011-06-17 07:01:31 +00:00
|
|
|
osd_device->ops.start_layer(osd_device,
|
|
|
|
layer->layer_info.id,
|
|
|
|
addr,
|
|
|
|
disp_obj->cbcr_ofst);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* interrupt service routine */
|
|
|
|
static irqreturn_t venc_isr(int irq, void *arg)
|
|
|
|
{
|
|
|
|
struct vpbe_display *disp_dev = (struct vpbe_display *)arg;
|
|
|
|
struct vpbe_layer *layer;
|
|
|
|
static unsigned last_event;
|
|
|
|
unsigned event = 0;
|
|
|
|
int fid;
|
|
|
|
int i;
|
|
|
|
|
2017-09-07 20:37:16 +00:00
|
|
|
if (!arg || !disp_dev->dev[0])
|
2011-06-17 07:01:31 +00:00
|
|
|
return IRQ_HANDLED;
|
|
|
|
|
|
|
|
if (venc_is_second_field(disp_dev))
|
|
|
|
event |= VENC_SECOND_FIELD;
|
|
|
|
else
|
|
|
|
event |= VENC_FIRST_FIELD;
|
|
|
|
|
|
|
|
if (event == (last_event & ~VENC_END_OF_FRAME)) {
|
|
|
|
/*
|
|
|
|
* If the display is non-interlaced, then we need to flag the
|
|
|
|
* end-of-frame event at every interrupt regardless of the
|
|
|
|
* value of the FIDST bit. We can conclude that the display is
|
|
|
|
* non-interlaced if the value of the FIDST bit is unchanged
|
|
|
|
* from the previous interrupt.
|
|
|
|
*/
|
|
|
|
event |= VENC_END_OF_FRAME;
|
|
|
|
} else if (event == VENC_SECOND_FIELD) {
|
|
|
|
/* end-of-frame for interlaced display */
|
|
|
|
event |= VENC_END_OF_FRAME;
|
|
|
|
}
|
|
|
|
last_event = event;
|
|
|
|
|
|
|
|
for (i = 0; i < VPBE_DISPLAY_MAX_DEVICES; i++) {
|
|
|
|
layer = disp_dev->dev[i];
|
2014-10-12 20:40:42 +00:00
|
|
|
|
|
|
|
if (!vb2_start_streaming_called(&layer->buffer_queue))
|
2011-06-17 07:01:31 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
if (layer->layer_first_int) {
|
|
|
|
layer->layer_first_int = 0;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
/* Check the field format */
|
|
|
|
if ((V4L2_FIELD_NONE == layer->pix_fmt.field) &&
|
|
|
|
(event & VENC_END_OF_FRAME)) {
|
|
|
|
/* Progressive mode */
|
|
|
|
|
|
|
|
vpbe_isr_even_field(disp_dev, layer);
|
|
|
|
vpbe_isr_odd_field(disp_dev, layer);
|
|
|
|
} else {
|
|
|
|
/* Interlaced mode */
|
|
|
|
|
|
|
|
layer->field_id ^= 1;
|
|
|
|
if (event & VENC_FIRST_FIELD)
|
|
|
|
fid = 0;
|
|
|
|
else
|
|
|
|
fid = 1;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If field id does not match with store
|
|
|
|
* field id
|
|
|
|
*/
|
|
|
|
if (fid != layer->field_id) {
|
|
|
|
/* Make them in sync */
|
|
|
|
layer->field_id = fid;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* device field id and local field id are
|
|
|
|
* in sync. If this is even field
|
|
|
|
*/
|
|
|
|
if (0 == fid)
|
|
|
|
vpbe_isr_even_field(disp_dev, layer);
|
|
|
|
else /* odd field */
|
|
|
|
vpbe_isr_odd_field(disp_dev, layer);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return IRQ_HANDLED;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* vpbe_buffer_prepare()
|
2012-10-22 12:27:13 +00:00
|
|
|
* This is the callback function called from vb2_qbuf() function
|
2011-06-17 07:01:31 +00:00
|
|
|
* the buffer is prepared and user space virtual address is converted into
|
|
|
|
* physical address
|
|
|
|
*/
|
2012-10-22 12:27:13 +00:00
|
|
|
static int vpbe_buffer_prepare(struct vb2_buffer *vb)
|
2011-06-17 07:01:31 +00:00
|
|
|
{
|
2012-10-22 12:27:13 +00:00
|
|
|
struct vb2_queue *q = vb->vb2_queue;
|
2014-10-12 20:40:31 +00:00
|
|
|
struct vpbe_layer *layer = vb2_get_drv_priv(q);
|
|
|
|
struct vpbe_device *vpbe_dev = layer->disp_dev->vpbe_dev;
|
2011-06-17 07:01:31 +00:00
|
|
|
unsigned long addr;
|
|
|
|
|
|
|
|
v4l2_dbg(1, debug, &vpbe_dev->v4l2_dev,
|
|
|
|
"vpbe_buffer_prepare\n");
|
|
|
|
|
2014-10-12 20:40:35 +00:00
|
|
|
vb2_set_plane_payload(vb, 0, layer->pix_fmt.sizeimage);
|
|
|
|
if (vb2_get_plane_payload(vb, 0) > vb2_plane_size(vb, 0))
|
|
|
|
return -EINVAL;
|
2011-06-17 07:01:31 +00:00
|
|
|
|
2014-10-12 20:40:35 +00:00
|
|
|
addr = vb2_dma_contig_plane_dma_addr(vb, 0);
|
|
|
|
if (!IS_ALIGNED(addr, 8)) {
|
|
|
|
v4l2_err(&vpbe_dev->v4l2_dev,
|
|
|
|
"buffer_prepare:offset is not aligned to 32 bytes\n");
|
|
|
|
return -EINVAL;
|
2011-06-17 07:01:31 +00:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* vpbe_buffer_setup()
|
|
|
|
* This function allocates memory for the buffers
|
|
|
|
*/
|
2012-10-22 12:27:13 +00:00
|
|
|
static int
|
2015-10-28 02:50:37 +00:00
|
|
|
vpbe_buffer_queue_setup(struct vb2_queue *vq,
|
2012-10-22 12:27:13 +00:00
|
|
|
unsigned int *nbuffers, unsigned int *nplanes,
|
2016-04-15 12:15:05 +00:00
|
|
|
unsigned int sizes[], struct device *alloc_devs[])
|
2012-10-22 12:27:13 +00:00
|
|
|
|
2011-06-17 07:01:31 +00:00
|
|
|
{
|
|
|
|
/* Get the file handle object and layer object */
|
2014-10-12 20:40:31 +00:00
|
|
|
struct vpbe_layer *layer = vb2_get_drv_priv(vq);
|
|
|
|
struct vpbe_device *vpbe_dev = layer->disp_dev->vpbe_dev;
|
2011-06-17 07:01:31 +00:00
|
|
|
|
|
|
|
v4l2_dbg(1, debug, &vpbe_dev->v4l2_dev, "vpbe_buffer_setup\n");
|
|
|
|
|
|
|
|
/* Store number of buffers allocated in numbuffer member */
|
2014-10-22 21:42:01 +00:00
|
|
|
if (vq->num_buffers + *nbuffers < VPBE_DEFAULT_NUM_BUFS)
|
|
|
|
*nbuffers = VPBE_DEFAULT_NUM_BUFS - vq->num_buffers;
|
2015-10-28 02:50:37 +00:00
|
|
|
|
|
|
|
if (*nplanes)
|
|
|
|
return sizes[0] < layer->pix_fmt.sizeimage ? -EINVAL : 0;
|
2012-10-22 12:27:13 +00:00
|
|
|
|
|
|
|
*nplanes = 1;
|
2015-10-28 02:50:37 +00:00
|
|
|
sizes[0] = layer->pix_fmt.sizeimage;
|
2011-06-17 07:01:31 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* vpbe_buffer_queue()
|
|
|
|
* This function adds the buffer to DMA queue
|
|
|
|
*/
|
2012-10-22 12:27:13 +00:00
|
|
|
static void vpbe_buffer_queue(struct vb2_buffer *vb)
|
2011-06-17 07:01:31 +00:00
|
|
|
{
|
[media] media: videobuf2: Restructure vb2_buffer
Remove v4l2 stuff - v4l2_buf, v4l2_plane - from struct vb2_buffer.
Add new member variables - bytesused, length, offset, userptr, fd,
data_offset - to struct vb2_plane in order to cover all information
of v4l2_plane.
struct vb2_plane {
<snip>
unsigned int bytesused;
unsigned int length;
union {
unsigned int offset;
unsigned long userptr;
int fd;
} m;
unsigned int data_offset;
}
Replace v4l2_buf with new member variables - index, type, memory - which
are common fields for buffer management.
struct vb2_buffer {
<snip>
unsigned int index;
unsigned int type;
unsigned int memory;
unsigned int num_planes;
struct vb2_plane planes[VIDEO_MAX_PLANES];
<snip>
};
v4l2 specific fields - flags, field, timestamp, timecode,
sequence - are moved to vb2_v4l2_buffer in videobuf2-v4l2.c
struct vb2_v4l2_buffer {
struct vb2_buffer vb2_buf;
__u32 flags;
__u32 field;
struct timeval timestamp;
struct v4l2_timecode timecode;
__u32 sequence;
};
Signed-off-by: Junghak Sung <jh1009.sung@samsung.com>
Signed-off-by: Geunyoung Kim <nenggun.kim@samsung.com>
Acked-by: Seung-Woo Kim <sw0312.kim@samsung.com>
Acked-by: Inki Dae <inki.dae@samsung.com>
Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
2015-09-22 13:30:30 +00:00
|
|
|
struct vb2_v4l2_buffer *vbuf = to_vb2_v4l2_buffer(vb);
|
2011-06-17 07:01:31 +00:00
|
|
|
/* Get the file handle object and layer object */
|
[media] media: videobuf2: Restructure vb2_buffer
Remove v4l2 stuff - v4l2_buf, v4l2_plane - from struct vb2_buffer.
Add new member variables - bytesused, length, offset, userptr, fd,
data_offset - to struct vb2_plane in order to cover all information
of v4l2_plane.
struct vb2_plane {
<snip>
unsigned int bytesused;
unsigned int length;
union {
unsigned int offset;
unsigned long userptr;
int fd;
} m;
unsigned int data_offset;
}
Replace v4l2_buf with new member variables - index, type, memory - which
are common fields for buffer management.
struct vb2_buffer {
<snip>
unsigned int index;
unsigned int type;
unsigned int memory;
unsigned int num_planes;
struct vb2_plane planes[VIDEO_MAX_PLANES];
<snip>
};
v4l2 specific fields - flags, field, timestamp, timecode,
sequence - are moved to vb2_v4l2_buffer in videobuf2-v4l2.c
struct vb2_v4l2_buffer {
struct vb2_buffer vb2_buf;
__u32 flags;
__u32 field;
struct timeval timestamp;
struct v4l2_timecode timecode;
__u32 sequence;
};
Signed-off-by: Junghak Sung <jh1009.sung@samsung.com>
Signed-off-by: Geunyoung Kim <nenggun.kim@samsung.com>
Acked-by: Seung-Woo Kim <sw0312.kim@samsung.com>
Acked-by: Inki Dae <inki.dae@samsung.com>
Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
2015-09-22 13:30:30 +00:00
|
|
|
struct vpbe_disp_buffer *buf = container_of(vbuf,
|
2012-10-22 12:27:13 +00:00
|
|
|
struct vpbe_disp_buffer, vb);
|
2014-10-12 20:40:31 +00:00
|
|
|
struct vpbe_layer *layer = vb2_get_drv_priv(vb->vb2_queue);
|
|
|
|
struct vpbe_display *disp = layer->disp_dev;
|
|
|
|
struct vpbe_device *vpbe_dev = layer->disp_dev->vpbe_dev;
|
2011-06-17 07:01:31 +00:00
|
|
|
unsigned long flags;
|
|
|
|
|
|
|
|
v4l2_dbg(1, debug, &vpbe_dev->v4l2_dev,
|
|
|
|
"vpbe_buffer_queue\n");
|
|
|
|
|
|
|
|
/* add the buffer to the DMA queue */
|
|
|
|
spin_lock_irqsave(&disp->dma_queue_lock, flags);
|
2012-10-22 12:27:13 +00:00
|
|
|
list_add_tail(&buf->list, &layer->dma_queue);
|
2011-06-17 07:01:31 +00:00
|
|
|
spin_unlock_irqrestore(&disp->dma_queue_lock, flags);
|
|
|
|
}
|
|
|
|
|
2012-10-22 12:27:13 +00:00
|
|
|
static int vpbe_start_streaming(struct vb2_queue *vq, unsigned int count)
|
|
|
|
{
|
2014-10-12 20:40:31 +00:00
|
|
|
struct vpbe_layer *layer = vb2_get_drv_priv(vq);
|
2014-10-12 20:40:38 +00:00
|
|
|
struct osd_state *osd_device = layer->disp_dev->osd_device;
|
2012-10-22 12:27:13 +00:00
|
|
|
int ret;
|
|
|
|
|
2018-04-06 12:24:27 +00:00
|
|
|
osd_device->ops.disable_layer(osd_device, layer->layer_info.id);
|
2014-10-12 20:40:38 +00:00
|
|
|
|
2012-10-22 12:27:13 +00:00
|
|
|
/* Get the next frame from the buffer queue */
|
|
|
|
layer->next_frm = layer->cur_frm = list_entry(layer->dma_queue.next,
|
|
|
|
struct vpbe_disp_buffer, list);
|
|
|
|
/* Remove buffer from the buffer queue */
|
|
|
|
list_del(&layer->cur_frm->list);
|
|
|
|
/* Mark state of the current frame to active */
|
[media] media: videobuf2: Restructure vb2_buffer
Remove v4l2 stuff - v4l2_buf, v4l2_plane - from struct vb2_buffer.
Add new member variables - bytesused, length, offset, userptr, fd,
data_offset - to struct vb2_plane in order to cover all information
of v4l2_plane.
struct vb2_plane {
<snip>
unsigned int bytesused;
unsigned int length;
union {
unsigned int offset;
unsigned long userptr;
int fd;
} m;
unsigned int data_offset;
}
Replace v4l2_buf with new member variables - index, type, memory - which
are common fields for buffer management.
struct vb2_buffer {
<snip>
unsigned int index;
unsigned int type;
unsigned int memory;
unsigned int num_planes;
struct vb2_plane planes[VIDEO_MAX_PLANES];
<snip>
};
v4l2 specific fields - flags, field, timestamp, timecode,
sequence - are moved to vb2_v4l2_buffer in videobuf2-v4l2.c
struct vb2_v4l2_buffer {
struct vb2_buffer vb2_buf;
__u32 flags;
__u32 field;
struct timeval timestamp;
struct v4l2_timecode timecode;
__u32 sequence;
};
Signed-off-by: Junghak Sung <jh1009.sung@samsung.com>
Signed-off-by: Geunyoung Kim <nenggun.kim@samsung.com>
Acked-by: Seung-Woo Kim <sw0312.kim@samsung.com>
Acked-by: Inki Dae <inki.dae@samsung.com>
Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
2015-09-22 13:30:30 +00:00
|
|
|
layer->cur_frm->vb.vb2_buf.state = VB2_BUF_STATE_ACTIVE;
|
2012-10-22 12:27:13 +00:00
|
|
|
/* Initialize field_id and started member */
|
|
|
|
layer->field_id = 0;
|
|
|
|
|
|
|
|
/* Set parameters in OSD and VENC */
|
2014-10-12 20:40:31 +00:00
|
|
|
ret = vpbe_set_osd_display_params(layer->disp_dev, layer);
|
2014-04-14 14:52:31 +00:00
|
|
|
if (ret < 0) {
|
|
|
|
struct vpbe_disp_buffer *buf, *tmp;
|
|
|
|
|
[media] media: videobuf2: Restructure vb2_buffer
Remove v4l2 stuff - v4l2_buf, v4l2_plane - from struct vb2_buffer.
Add new member variables - bytesused, length, offset, userptr, fd,
data_offset - to struct vb2_plane in order to cover all information
of v4l2_plane.
struct vb2_plane {
<snip>
unsigned int bytesused;
unsigned int length;
union {
unsigned int offset;
unsigned long userptr;
int fd;
} m;
unsigned int data_offset;
}
Replace v4l2_buf with new member variables - index, type, memory - which
are common fields for buffer management.
struct vb2_buffer {
<snip>
unsigned int index;
unsigned int type;
unsigned int memory;
unsigned int num_planes;
struct vb2_plane planes[VIDEO_MAX_PLANES];
<snip>
};
v4l2 specific fields - flags, field, timestamp, timecode,
sequence - are moved to vb2_v4l2_buffer in videobuf2-v4l2.c
struct vb2_v4l2_buffer {
struct vb2_buffer vb2_buf;
__u32 flags;
__u32 field;
struct timeval timestamp;
struct v4l2_timecode timecode;
__u32 sequence;
};
Signed-off-by: Junghak Sung <jh1009.sung@samsung.com>
Signed-off-by: Geunyoung Kim <nenggun.kim@samsung.com>
Acked-by: Seung-Woo Kim <sw0312.kim@samsung.com>
Acked-by: Inki Dae <inki.dae@samsung.com>
Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
2015-09-22 13:30:30 +00:00
|
|
|
vb2_buffer_done(&layer->cur_frm->vb.vb2_buf,
|
|
|
|
VB2_BUF_STATE_QUEUED);
|
2014-04-14 14:52:31 +00:00
|
|
|
list_for_each_entry_safe(buf, tmp, &layer->dma_queue, list) {
|
|
|
|
list_del(&buf->list);
|
[media] media: videobuf2: Restructure vb2_buffer
Remove v4l2 stuff - v4l2_buf, v4l2_plane - from struct vb2_buffer.
Add new member variables - bytesused, length, offset, userptr, fd,
data_offset - to struct vb2_plane in order to cover all information
of v4l2_plane.
struct vb2_plane {
<snip>
unsigned int bytesused;
unsigned int length;
union {
unsigned int offset;
unsigned long userptr;
int fd;
} m;
unsigned int data_offset;
}
Replace v4l2_buf with new member variables - index, type, memory - which
are common fields for buffer management.
struct vb2_buffer {
<snip>
unsigned int index;
unsigned int type;
unsigned int memory;
unsigned int num_planes;
struct vb2_plane planes[VIDEO_MAX_PLANES];
<snip>
};
v4l2 specific fields - flags, field, timestamp, timecode,
sequence - are moved to vb2_v4l2_buffer in videobuf2-v4l2.c
struct vb2_v4l2_buffer {
struct vb2_buffer vb2_buf;
__u32 flags;
__u32 field;
struct timeval timestamp;
struct v4l2_timecode timecode;
__u32 sequence;
};
Signed-off-by: Junghak Sung <jh1009.sung@samsung.com>
Signed-off-by: Geunyoung Kim <nenggun.kim@samsung.com>
Acked-by: Seung-Woo Kim <sw0312.kim@samsung.com>
Acked-by: Inki Dae <inki.dae@samsung.com>
Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
2015-09-22 13:30:30 +00:00
|
|
|
vb2_buffer_done(&buf->vb.vb2_buf,
|
|
|
|
VB2_BUF_STATE_QUEUED);
|
2014-04-14 14:52:31 +00:00
|
|
|
}
|
|
|
|
|
2012-10-22 12:27:13 +00:00
|
|
|
return ret;
|
2014-04-14 14:52:31 +00:00
|
|
|
}
|
2012-10-22 12:27:13 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* if request format is yuv420 semiplanar, need to
|
|
|
|
* enable both video windows
|
|
|
|
*/
|
|
|
|
layer->layer_first_int = 1;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2014-04-17 05:47:21 +00:00
|
|
|
static void vpbe_stop_streaming(struct vb2_queue *vq)
|
2012-10-22 12:27:13 +00:00
|
|
|
{
|
2014-10-12 20:40:31 +00:00
|
|
|
struct vpbe_layer *layer = vb2_get_drv_priv(vq);
|
2014-10-12 20:40:38 +00:00
|
|
|
struct osd_state *osd_device = layer->disp_dev->osd_device;
|
2014-10-12 20:40:31 +00:00
|
|
|
struct vpbe_display *disp = layer->disp_dev;
|
2014-03-22 11:03:09 +00:00
|
|
|
unsigned long flags;
|
2012-10-22 12:27:13 +00:00
|
|
|
|
|
|
|
if (!vb2_is_streaming(vq))
|
2014-04-17 05:47:21 +00:00
|
|
|
return;
|
2012-10-22 12:27:13 +00:00
|
|
|
|
2014-10-12 20:40:38 +00:00
|
|
|
osd_device->ops.disable_layer(osd_device, layer->layer_info.id);
|
|
|
|
|
2012-10-22 12:27:13 +00:00
|
|
|
/* release all active buffers */
|
2014-03-22 11:03:09 +00:00
|
|
|
spin_lock_irqsave(&disp->dma_queue_lock, flags);
|
|
|
|
if (layer->cur_frm == layer->next_frm) {
|
[media] media: videobuf2: Restructure vb2_buffer
Remove v4l2 stuff - v4l2_buf, v4l2_plane - from struct vb2_buffer.
Add new member variables - bytesused, length, offset, userptr, fd,
data_offset - to struct vb2_plane in order to cover all information
of v4l2_plane.
struct vb2_plane {
<snip>
unsigned int bytesused;
unsigned int length;
union {
unsigned int offset;
unsigned long userptr;
int fd;
} m;
unsigned int data_offset;
}
Replace v4l2_buf with new member variables - index, type, memory - which
are common fields for buffer management.
struct vb2_buffer {
<snip>
unsigned int index;
unsigned int type;
unsigned int memory;
unsigned int num_planes;
struct vb2_plane planes[VIDEO_MAX_PLANES];
<snip>
};
v4l2 specific fields - flags, field, timestamp, timecode,
sequence - are moved to vb2_v4l2_buffer in videobuf2-v4l2.c
struct vb2_v4l2_buffer {
struct vb2_buffer vb2_buf;
__u32 flags;
__u32 field;
struct timeval timestamp;
struct v4l2_timecode timecode;
__u32 sequence;
};
Signed-off-by: Junghak Sung <jh1009.sung@samsung.com>
Signed-off-by: Geunyoung Kim <nenggun.kim@samsung.com>
Acked-by: Seung-Woo Kim <sw0312.kim@samsung.com>
Acked-by: Inki Dae <inki.dae@samsung.com>
Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
2015-09-22 13:30:30 +00:00
|
|
|
vb2_buffer_done(&layer->cur_frm->vb.vb2_buf,
|
|
|
|
VB2_BUF_STATE_ERROR);
|
2014-03-22 11:03:09 +00:00
|
|
|
} else {
|
2017-09-07 20:37:16 +00:00
|
|
|
if (layer->cur_frm)
|
[media] media: videobuf2: Restructure vb2_buffer
Remove v4l2 stuff - v4l2_buf, v4l2_plane - from struct vb2_buffer.
Add new member variables - bytesused, length, offset, userptr, fd,
data_offset - to struct vb2_plane in order to cover all information
of v4l2_plane.
struct vb2_plane {
<snip>
unsigned int bytesused;
unsigned int length;
union {
unsigned int offset;
unsigned long userptr;
int fd;
} m;
unsigned int data_offset;
}
Replace v4l2_buf with new member variables - index, type, memory - which
are common fields for buffer management.
struct vb2_buffer {
<snip>
unsigned int index;
unsigned int type;
unsigned int memory;
unsigned int num_planes;
struct vb2_plane planes[VIDEO_MAX_PLANES];
<snip>
};
v4l2 specific fields - flags, field, timestamp, timecode,
sequence - are moved to vb2_v4l2_buffer in videobuf2-v4l2.c
struct vb2_v4l2_buffer {
struct vb2_buffer vb2_buf;
__u32 flags;
__u32 field;
struct timeval timestamp;
struct v4l2_timecode timecode;
__u32 sequence;
};
Signed-off-by: Junghak Sung <jh1009.sung@samsung.com>
Signed-off-by: Geunyoung Kim <nenggun.kim@samsung.com>
Acked-by: Seung-Woo Kim <sw0312.kim@samsung.com>
Acked-by: Inki Dae <inki.dae@samsung.com>
Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
2015-09-22 13:30:30 +00:00
|
|
|
vb2_buffer_done(&layer->cur_frm->vb.vb2_buf,
|
2014-03-22 11:03:09 +00:00
|
|
|
VB2_BUF_STATE_ERROR);
|
2017-09-07 20:37:16 +00:00
|
|
|
if (layer->next_frm)
|
[media] media: videobuf2: Restructure vb2_buffer
Remove v4l2 stuff - v4l2_buf, v4l2_plane - from struct vb2_buffer.
Add new member variables - bytesused, length, offset, userptr, fd,
data_offset - to struct vb2_plane in order to cover all information
of v4l2_plane.
struct vb2_plane {
<snip>
unsigned int bytesused;
unsigned int length;
union {
unsigned int offset;
unsigned long userptr;
int fd;
} m;
unsigned int data_offset;
}
Replace v4l2_buf with new member variables - index, type, memory - which
are common fields for buffer management.
struct vb2_buffer {
<snip>
unsigned int index;
unsigned int type;
unsigned int memory;
unsigned int num_planes;
struct vb2_plane planes[VIDEO_MAX_PLANES];
<snip>
};
v4l2 specific fields - flags, field, timestamp, timecode,
sequence - are moved to vb2_v4l2_buffer in videobuf2-v4l2.c
struct vb2_v4l2_buffer {
struct vb2_buffer vb2_buf;
__u32 flags;
__u32 field;
struct timeval timestamp;
struct v4l2_timecode timecode;
__u32 sequence;
};
Signed-off-by: Junghak Sung <jh1009.sung@samsung.com>
Signed-off-by: Geunyoung Kim <nenggun.kim@samsung.com>
Acked-by: Seung-Woo Kim <sw0312.kim@samsung.com>
Acked-by: Inki Dae <inki.dae@samsung.com>
Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
2015-09-22 13:30:30 +00:00
|
|
|
vb2_buffer_done(&layer->next_frm->vb.vb2_buf,
|
2014-03-22 11:03:09 +00:00
|
|
|
VB2_BUF_STATE_ERROR);
|
|
|
|
}
|
|
|
|
|
2012-10-22 12:27:13 +00:00
|
|
|
while (!list_empty(&layer->dma_queue)) {
|
|
|
|
layer->next_frm = list_entry(layer->dma_queue.next,
|
|
|
|
struct vpbe_disp_buffer, list);
|
|
|
|
list_del(&layer->next_frm->list);
|
[media] media: videobuf2: Restructure vb2_buffer
Remove v4l2 stuff - v4l2_buf, v4l2_plane - from struct vb2_buffer.
Add new member variables - bytesused, length, offset, userptr, fd,
data_offset - to struct vb2_plane in order to cover all information
of v4l2_plane.
struct vb2_plane {
<snip>
unsigned int bytesused;
unsigned int length;
union {
unsigned int offset;
unsigned long userptr;
int fd;
} m;
unsigned int data_offset;
}
Replace v4l2_buf with new member variables - index, type, memory - which
are common fields for buffer management.
struct vb2_buffer {
<snip>
unsigned int index;
unsigned int type;
unsigned int memory;
unsigned int num_planes;
struct vb2_plane planes[VIDEO_MAX_PLANES];
<snip>
};
v4l2 specific fields - flags, field, timestamp, timecode,
sequence - are moved to vb2_v4l2_buffer in videobuf2-v4l2.c
struct vb2_v4l2_buffer {
struct vb2_buffer vb2_buf;
__u32 flags;
__u32 field;
struct timeval timestamp;
struct v4l2_timecode timecode;
__u32 sequence;
};
Signed-off-by: Junghak Sung <jh1009.sung@samsung.com>
Signed-off-by: Geunyoung Kim <nenggun.kim@samsung.com>
Acked-by: Seung-Woo Kim <sw0312.kim@samsung.com>
Acked-by: Inki Dae <inki.dae@samsung.com>
Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
2015-09-22 13:30:30 +00:00
|
|
|
vb2_buffer_done(&layer->next_frm->vb.vb2_buf,
|
|
|
|
VB2_BUF_STATE_ERROR);
|
2012-10-22 12:27:13 +00:00
|
|
|
}
|
2014-03-22 11:03:09 +00:00
|
|
|
spin_unlock_irqrestore(&disp->dma_queue_lock, flags);
|
2012-10-22 12:27:13 +00:00
|
|
|
}
|
|
|
|
|
2017-08-05 10:47:09 +00:00
|
|
|
static const struct vb2_ops video_qops = {
|
2012-10-22 12:27:13 +00:00
|
|
|
.queue_setup = vpbe_buffer_queue_setup,
|
2014-10-12 20:40:33 +00:00
|
|
|
.wait_prepare = vb2_ops_wait_prepare,
|
|
|
|
.wait_finish = vb2_ops_wait_finish,
|
2011-06-17 07:01:31 +00:00
|
|
|
.buf_prepare = vpbe_buffer_prepare,
|
2012-10-22 12:27:13 +00:00
|
|
|
.start_streaming = vpbe_start_streaming,
|
|
|
|
.stop_streaming = vpbe_stop_streaming,
|
2011-06-17 07:01:31 +00:00
|
|
|
.buf_queue = vpbe_buffer_queue,
|
|
|
|
};
|
|
|
|
|
|
|
|
static
|
|
|
|
struct vpbe_layer*
|
|
|
|
_vpbe_display_get_other_win_layer(struct vpbe_display *disp_dev,
|
|
|
|
struct vpbe_layer *layer)
|
|
|
|
{
|
|
|
|
enum vpbe_display_device_id thiswin, otherwin;
|
|
|
|
thiswin = layer->device_id;
|
|
|
|
|
|
|
|
otherwin = (thiswin == VPBE_DISPLAY_DEVICE_0) ?
|
|
|
|
VPBE_DISPLAY_DEVICE_1 : VPBE_DISPLAY_DEVICE_0;
|
|
|
|
return disp_dev->dev[otherwin];
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vpbe_set_osd_display_params(struct vpbe_display *disp_dev,
|
|
|
|
struct vpbe_layer *layer)
|
|
|
|
{
|
|
|
|
struct osd_layer_config *cfg = &layer->layer_info.config;
|
|
|
|
struct osd_state *osd_device = disp_dev->osd_device;
|
|
|
|
struct vpbe_device *vpbe_dev = disp_dev->vpbe_dev;
|
|
|
|
unsigned long addr;
|
|
|
|
int ret;
|
|
|
|
|
[media] media: videobuf2: Restructure vb2_buffer
Remove v4l2 stuff - v4l2_buf, v4l2_plane - from struct vb2_buffer.
Add new member variables - bytesused, length, offset, userptr, fd,
data_offset - to struct vb2_plane in order to cover all information
of v4l2_plane.
struct vb2_plane {
<snip>
unsigned int bytesused;
unsigned int length;
union {
unsigned int offset;
unsigned long userptr;
int fd;
} m;
unsigned int data_offset;
}
Replace v4l2_buf with new member variables - index, type, memory - which
are common fields for buffer management.
struct vb2_buffer {
<snip>
unsigned int index;
unsigned int type;
unsigned int memory;
unsigned int num_planes;
struct vb2_plane planes[VIDEO_MAX_PLANES];
<snip>
};
v4l2 specific fields - flags, field, timestamp, timecode,
sequence - are moved to vb2_v4l2_buffer in videobuf2-v4l2.c
struct vb2_v4l2_buffer {
struct vb2_buffer vb2_buf;
__u32 flags;
__u32 field;
struct timeval timestamp;
struct v4l2_timecode timecode;
__u32 sequence;
};
Signed-off-by: Junghak Sung <jh1009.sung@samsung.com>
Signed-off-by: Geunyoung Kim <nenggun.kim@samsung.com>
Acked-by: Seung-Woo Kim <sw0312.kim@samsung.com>
Acked-by: Inki Dae <inki.dae@samsung.com>
Signed-off-by: Hans Verkuil <hans.verkuil@cisco.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
2015-09-22 13:30:30 +00:00
|
|
|
addr = vb2_dma_contig_plane_dma_addr(&layer->cur_frm->vb.vb2_buf, 0);
|
2011-06-17 07:01:31 +00:00
|
|
|
/* Set address in the display registers */
|
|
|
|
osd_device->ops.start_layer(osd_device,
|
|
|
|
layer->layer_info.id,
|
|
|
|
addr,
|
|
|
|
disp_dev->cbcr_ofst);
|
|
|
|
|
|
|
|
ret = osd_device->ops.enable_layer(osd_device,
|
|
|
|
layer->layer_info.id, 0);
|
|
|
|
if (ret < 0) {
|
|
|
|
v4l2_err(&vpbe_dev->v4l2_dev,
|
|
|
|
"Error in enabling osd window layer 0\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Enable the window */
|
|
|
|
layer->layer_info.enable = 1;
|
|
|
|
if (cfg->pixfmt == PIXFMT_NV12) {
|
|
|
|
struct vpbe_layer *otherlayer =
|
|
|
|
_vpbe_display_get_other_win_layer(disp_dev, layer);
|
|
|
|
|
|
|
|
ret = osd_device->ops.enable_layer(osd_device,
|
|
|
|
otherlayer->layer_info.id, 1);
|
|
|
|
if (ret < 0) {
|
|
|
|
v4l2_err(&vpbe_dev->v4l2_dev,
|
|
|
|
"Error in enabling osd window layer 1\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
otherlayer->layer_info.enable = 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
vpbe_disp_calculate_scale_factor(struct vpbe_display *disp_dev,
|
|
|
|
struct vpbe_layer *layer,
|
|
|
|
int expected_xsize, int expected_ysize)
|
|
|
|
{
|
|
|
|
struct display_layer_info *layer_info = &layer->layer_info;
|
|
|
|
struct v4l2_pix_format *pixfmt = &layer->pix_fmt;
|
|
|
|
struct osd_layer_config *cfg = &layer->layer_info.config;
|
|
|
|
struct vpbe_device *vpbe_dev = disp_dev->vpbe_dev;
|
|
|
|
int calculated_xsize;
|
|
|
|
int h_exp = 0;
|
|
|
|
int v_exp = 0;
|
|
|
|
int h_scale;
|
|
|
|
int v_scale;
|
|
|
|
|
2012-10-01 14:39:46 +00:00
|
|
|
v4l2_std_id standard_id = vpbe_dev->current_timings.std_id;
|
2011-06-17 07:01:31 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Application initially set the image format. Current display
|
|
|
|
* size is obtained from the vpbe display controller. expected_xsize
|
2016-07-03 10:53:31 +00:00
|
|
|
* and expected_ysize are set through S_SELECTION ioctl. Based on this,
|
2011-06-17 07:01:31 +00:00
|
|
|
* driver will calculate the scale factors for vertical and
|
|
|
|
* horizontal direction so that the image is displayed scaled
|
|
|
|
* and expanded. Application uses expansion to display the image
|
|
|
|
* in a square pixel. Otherwise it is displayed using displays
|
|
|
|
* pixel aspect ratio.It is expected that application chooses
|
|
|
|
* the crop coordinates for cropped or scaled display. if crop
|
|
|
|
* size is less than the image size, it is displayed cropped or
|
|
|
|
* it is displayed scaled and/or expanded.
|
|
|
|
*
|
|
|
|
* to begin with, set the crop window same as expected. Later we
|
|
|
|
* will override with scaled window size
|
|
|
|
*/
|
|
|
|
|
|
|
|
cfg->xsize = pixfmt->width;
|
|
|
|
cfg->ysize = pixfmt->height;
|
|
|
|
layer_info->h_zoom = ZOOM_X1; /* no horizontal zoom */
|
|
|
|
layer_info->v_zoom = ZOOM_X1; /* no horizontal zoom */
|
|
|
|
layer_info->h_exp = H_EXP_OFF; /* no horizontal zoom */
|
|
|
|
layer_info->v_exp = V_EXP_OFF; /* no horizontal zoom */
|
|
|
|
|
|
|
|
if (pixfmt->width < expected_xsize) {
|
|
|
|
h_scale = vpbe_dev->current_timings.xres / pixfmt->width;
|
|
|
|
if (h_scale < 2)
|
|
|
|
h_scale = 1;
|
|
|
|
else if (h_scale >= 4)
|
|
|
|
h_scale = 4;
|
|
|
|
else
|
|
|
|
h_scale = 2;
|
|
|
|
cfg->xsize *= h_scale;
|
|
|
|
if (cfg->xsize < expected_xsize) {
|
|
|
|
if ((standard_id & V4L2_STD_525_60) ||
|
|
|
|
(standard_id & V4L2_STD_625_50)) {
|
|
|
|
calculated_xsize = (cfg->xsize *
|
|
|
|
VPBE_DISPLAY_H_EXP_RATIO_N) /
|
|
|
|
VPBE_DISPLAY_H_EXP_RATIO_D;
|
|
|
|
if (calculated_xsize <= expected_xsize) {
|
|
|
|
h_exp = 1;
|
|
|
|
cfg->xsize = calculated_xsize;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (h_scale == 2)
|
|
|
|
layer_info->h_zoom = ZOOM_X2;
|
|
|
|
else if (h_scale == 4)
|
|
|
|
layer_info->h_zoom = ZOOM_X4;
|
|
|
|
if (h_exp)
|
|
|
|
layer_info->h_exp = H_EXP_9_OVER_8;
|
|
|
|
} else {
|
|
|
|
/* no scaling, only cropping. Set display area to crop area */
|
|
|
|
cfg->xsize = expected_xsize;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pixfmt->height < expected_ysize) {
|
|
|
|
v_scale = expected_ysize / pixfmt->height;
|
|
|
|
if (v_scale < 2)
|
|
|
|
v_scale = 1;
|
|
|
|
else if (v_scale >= 4)
|
|
|
|
v_scale = 4;
|
|
|
|
else
|
|
|
|
v_scale = 2;
|
|
|
|
cfg->ysize *= v_scale;
|
|
|
|
if (cfg->ysize < expected_ysize) {
|
|
|
|
if ((standard_id & V4L2_STD_625_50)) {
|
|
|
|
calculated_xsize = (cfg->ysize *
|
|
|
|
VPBE_DISPLAY_V_EXP_RATIO_N) /
|
|
|
|
VPBE_DISPLAY_V_EXP_RATIO_D;
|
|
|
|
if (calculated_xsize <= expected_ysize) {
|
|
|
|
v_exp = 1;
|
|
|
|
cfg->ysize = calculated_xsize;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (v_scale == 2)
|
|
|
|
layer_info->v_zoom = ZOOM_X2;
|
|
|
|
else if (v_scale == 4)
|
|
|
|
layer_info->v_zoom = ZOOM_X4;
|
|
|
|
if (v_exp)
|
2018-09-15 06:16:15 +00:00
|
|
|
layer_info->v_exp = V_EXP_6_OVER_5;
|
2011-06-17 07:01:31 +00:00
|
|
|
} else {
|
|
|
|
/* no scaling, only cropping. Set display area to crop area */
|
|
|
|
cfg->ysize = expected_ysize;
|
|
|
|
}
|
|
|
|
v4l2_dbg(1, debug, &vpbe_dev->v4l2_dev,
|
|
|
|
"crop display xsize = %d, ysize = %d\n",
|
|
|
|
cfg->xsize, cfg->ysize);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void vpbe_disp_adj_position(struct vpbe_display *disp_dev,
|
|
|
|
struct vpbe_layer *layer,
|
|
|
|
int top, int left)
|
|
|
|
{
|
|
|
|
struct osd_layer_config *cfg = &layer->layer_info.config;
|
|
|
|
struct vpbe_device *vpbe_dev = disp_dev->vpbe_dev;
|
|
|
|
|
|
|
|
cfg->xpos = min((unsigned int)left,
|
|
|
|
vpbe_dev->current_timings.xres - cfg->xsize);
|
|
|
|
cfg->ypos = min((unsigned int)top,
|
|
|
|
vpbe_dev->current_timings.yres - cfg->ysize);
|
|
|
|
|
|
|
|
v4l2_dbg(1, debug, &vpbe_dev->v4l2_dev,
|
|
|
|
"new xpos = %d, ypos = %d\n",
|
|
|
|
cfg->xpos, cfg->ypos);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void vpbe_disp_check_window_params(struct vpbe_display *disp_dev,
|
|
|
|
struct v4l2_rect *c)
|
|
|
|
{
|
|
|
|
struct vpbe_device *vpbe_dev = disp_dev->vpbe_dev;
|
|
|
|
|
|
|
|
if ((c->width == 0) ||
|
|
|
|
((c->width + c->left) > vpbe_dev->current_timings.xres))
|
|
|
|
c->width = vpbe_dev->current_timings.xres - c->left;
|
|
|
|
|
|
|
|
if ((c->height == 0) || ((c->height + c->top) >
|
|
|
|
vpbe_dev->current_timings.yres))
|
|
|
|
c->height = vpbe_dev->current_timings.yres - c->top;
|
|
|
|
|
|
|
|
/* window height must be even for interlaced display */
|
|
|
|
if (vpbe_dev->current_timings.interlaced)
|
|
|
|
c->height &= (~0x01);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2018-04-05 16:51:00 +00:00
|
|
|
/*
|
2011-06-17 07:01:31 +00:00
|
|
|
* vpbe_try_format()
|
|
|
|
* If user application provides width and height, and have bytesperline set
|
|
|
|
* to zero, driver calculates bytesperline and sizeimage based on hardware
|
|
|
|
* limits.
|
|
|
|
*/
|
|
|
|
static int vpbe_try_format(struct vpbe_display *disp_dev,
|
|
|
|
struct v4l2_pix_format *pixfmt, int check)
|
|
|
|
{
|
|
|
|
struct vpbe_device *vpbe_dev = disp_dev->vpbe_dev;
|
|
|
|
int min_height = 1;
|
|
|
|
int min_width = 32;
|
|
|
|
int max_height;
|
|
|
|
int max_width;
|
|
|
|
int bpp;
|
|
|
|
|
|
|
|
if ((pixfmt->pixelformat != V4L2_PIX_FMT_UYVY) &&
|
|
|
|
(pixfmt->pixelformat != V4L2_PIX_FMT_NV12))
|
|
|
|
/* choose default as V4L2_PIX_FMT_UYVY */
|
|
|
|
pixfmt->pixelformat = V4L2_PIX_FMT_UYVY;
|
|
|
|
|
|
|
|
/* Check the field format */
|
|
|
|
if ((pixfmt->field != V4L2_FIELD_INTERLACED) &&
|
|
|
|
(pixfmt->field != V4L2_FIELD_NONE)) {
|
|
|
|
if (vpbe_dev->current_timings.interlaced)
|
|
|
|
pixfmt->field = V4L2_FIELD_INTERLACED;
|
|
|
|
else
|
|
|
|
pixfmt->field = V4L2_FIELD_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pixfmt->field == V4L2_FIELD_INTERLACED)
|
|
|
|
min_height = 2;
|
|
|
|
|
|
|
|
if (pixfmt->pixelformat == V4L2_PIX_FMT_NV12)
|
|
|
|
bpp = 1;
|
|
|
|
else
|
|
|
|
bpp = 2;
|
|
|
|
|
|
|
|
max_width = vpbe_dev->current_timings.xres;
|
|
|
|
max_height = vpbe_dev->current_timings.yres;
|
|
|
|
|
|
|
|
min_width /= bpp;
|
|
|
|
|
|
|
|
if (!pixfmt->width || (pixfmt->width < min_width) ||
|
|
|
|
(pixfmt->width > max_width)) {
|
|
|
|
pixfmt->width = vpbe_dev->current_timings.xres;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!pixfmt->height || (pixfmt->height < min_height) ||
|
|
|
|
(pixfmt->height > max_height)) {
|
|
|
|
pixfmt->height = vpbe_dev->current_timings.yres;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pixfmt->bytesperline < (pixfmt->width * bpp))
|
|
|
|
pixfmt->bytesperline = pixfmt->width * bpp;
|
|
|
|
|
|
|
|
/* Make the bytesperline 32 byte aligned */
|
|
|
|
pixfmt->bytesperline = ((pixfmt->width * bpp + 31) & ~31);
|
|
|
|
|
|
|
|
if (pixfmt->pixelformat == V4L2_PIX_FMT_NV12)
|
|
|
|
pixfmt->sizeimage = pixfmt->bytesperline * pixfmt->height +
|
|
|
|
(pixfmt->bytesperline * pixfmt->height >> 1);
|
|
|
|
else
|
|
|
|
pixfmt->sizeimage = pixfmt->bytesperline * pixfmt->height;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vpbe_display_querycap(struct file *file, void *priv,
|
|
|
|
struct v4l2_capability *cap)
|
|
|
|
{
|
2014-10-12 20:40:37 +00:00
|
|
|
struct vpbe_layer *layer = video_drvdata(file);
|
|
|
|
struct vpbe_device *vpbe_dev = layer->disp_dev->vpbe_dev;
|
2011-06-17 07:01:31 +00:00
|
|
|
|
2012-10-22 12:27:14 +00:00
|
|
|
cap->device_caps = V4L2_CAP_VIDEO_OUTPUT | V4L2_CAP_STREAMING;
|
|
|
|
cap->capabilities = cap->device_caps | V4L2_CAP_DEVICE_CAPS;
|
|
|
|
snprintf(cap->driver, sizeof(cap->driver), "%s",
|
|
|
|
dev_name(vpbe_dev->pdev));
|
|
|
|
snprintf(cap->bus_info, sizeof(cap->bus_info), "platform:%s",
|
|
|
|
dev_name(vpbe_dev->pdev));
|
2018-09-10 12:19:14 +00:00
|
|
|
strscpy(cap->card, vpbe_dev->cfg->module_name, sizeof(cap->card));
|
2011-06-17 07:01:31 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-07-03 10:53:31 +00:00
|
|
|
static int vpbe_display_s_selection(struct file *file, void *priv,
|
|
|
|
struct v4l2_selection *sel)
|
2011-06-17 07:01:31 +00:00
|
|
|
{
|
2014-10-12 20:40:37 +00:00
|
|
|
struct vpbe_layer *layer = video_drvdata(file);
|
|
|
|
struct vpbe_display *disp_dev = layer->disp_dev;
|
2011-06-17 07:01:31 +00:00
|
|
|
struct vpbe_device *vpbe_dev = disp_dev->vpbe_dev;
|
|
|
|
struct osd_layer_config *cfg = &layer->layer_info.config;
|
|
|
|
struct osd_state *osd_device = disp_dev->osd_device;
|
2016-07-03 10:53:31 +00:00
|
|
|
struct v4l2_rect rect = sel->r;
|
2011-06-17 07:01:31 +00:00
|
|
|
int ret;
|
|
|
|
|
|
|
|
v4l2_dbg(1, debug, &vpbe_dev->v4l2_dev,
|
2016-07-03 10:53:31 +00:00
|
|
|
"VIDIOC_S_SELECTION, layer id = %d\n", layer->device_id);
|
2011-06-17 07:01:31 +00:00
|
|
|
|
2016-07-03 10:53:31 +00:00
|
|
|
if (sel->type != V4L2_BUF_TYPE_VIDEO_OUTPUT ||
|
|
|
|
sel->target != V4L2_SEL_TGT_CROP)
|
2011-06-17 07:01:31 +00:00
|
|
|
return -EINVAL;
|
|
|
|
|
2012-10-03 05:13:28 +00:00
|
|
|
if (rect.top < 0)
|
|
|
|
rect.top = 0;
|
|
|
|
if (rect.left < 0)
|
|
|
|
rect.left = 0;
|
2011-06-17 07:01:31 +00:00
|
|
|
|
2012-10-03 05:13:28 +00:00
|
|
|
vpbe_disp_check_window_params(disp_dev, &rect);
|
2011-06-17 07:01:31 +00:00
|
|
|
|
|
|
|
osd_device->ops.get_layer_config(osd_device,
|
|
|
|
layer->layer_info.id, cfg);
|
|
|
|
|
|
|
|
vpbe_disp_calculate_scale_factor(disp_dev, layer,
|
2012-10-03 05:13:28 +00:00
|
|
|
rect.width,
|
|
|
|
rect.height);
|
|
|
|
vpbe_disp_adj_position(disp_dev, layer, rect.top,
|
|
|
|
rect.left);
|
2011-06-17 07:01:31 +00:00
|
|
|
ret = osd_device->ops.set_layer_config(osd_device,
|
|
|
|
layer->layer_info.id, cfg);
|
|
|
|
if (ret < 0) {
|
|
|
|
v4l2_err(&vpbe_dev->v4l2_dev,
|
|
|
|
"Error in set layer config:\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* apply zooming and h or v expansion */
|
|
|
|
osd_device->ops.set_zoom(osd_device,
|
|
|
|
layer->layer_info.id,
|
|
|
|
layer->layer_info.h_zoom,
|
|
|
|
layer->layer_info.v_zoom);
|
|
|
|
ret = osd_device->ops.set_vid_expansion(osd_device,
|
|
|
|
layer->layer_info.h_exp,
|
|
|
|
layer->layer_info.v_exp);
|
|
|
|
if (ret < 0) {
|
|
|
|
v4l2_err(&vpbe_dev->v4l2_dev,
|
|
|
|
"Error in set vid expansion:\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((layer->layer_info.h_zoom != ZOOM_X1) ||
|
|
|
|
(layer->layer_info.v_zoom != ZOOM_X1) ||
|
|
|
|
(layer->layer_info.h_exp != H_EXP_OFF) ||
|
|
|
|
(layer->layer_info.v_exp != V_EXP_OFF))
|
|
|
|
/* Enable expansion filter */
|
|
|
|
osd_device->ops.set_interpolation_filter(osd_device, 1);
|
|
|
|
else
|
|
|
|
osd_device->ops.set_interpolation_filter(osd_device, 0);
|
|
|
|
|
2016-07-03 10:53:31 +00:00
|
|
|
sel->r = rect;
|
2011-06-17 07:01:31 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-07-03 10:53:31 +00:00
|
|
|
static int vpbe_display_g_selection(struct file *file, void *priv,
|
|
|
|
struct v4l2_selection *sel)
|
2011-06-17 07:01:31 +00:00
|
|
|
{
|
2014-10-12 20:40:37 +00:00
|
|
|
struct vpbe_layer *layer = video_drvdata(file);
|
2011-06-17 07:01:31 +00:00
|
|
|
struct osd_layer_config *cfg = &layer->layer_info.config;
|
2014-10-12 20:40:37 +00:00
|
|
|
struct vpbe_device *vpbe_dev = layer->disp_dev->vpbe_dev;
|
|
|
|
struct osd_state *osd_device = layer->disp_dev->osd_device;
|
2016-07-03 10:53:31 +00:00
|
|
|
struct v4l2_rect *rect = &sel->r;
|
2011-06-17 07:01:31 +00:00
|
|
|
|
|
|
|
v4l2_dbg(1, debug, &vpbe_dev->v4l2_dev,
|
2016-07-03 10:53:31 +00:00
|
|
|
"VIDIOC_G_SELECTION, layer id = %d\n",
|
2011-06-17 07:01:31 +00:00
|
|
|
layer->device_id);
|
|
|
|
|
2016-07-03 10:53:31 +00:00
|
|
|
if (sel->type != V4L2_BUF_TYPE_VIDEO_OUTPUT)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
switch (sel->target) {
|
|
|
|
case V4L2_SEL_TGT_CROP:
|
|
|
|
osd_device->ops.get_layer_config(osd_device,
|
|
|
|
layer->layer_info.id, cfg);
|
|
|
|
rect->top = cfg->ypos;
|
|
|
|
rect->left = cfg->xpos;
|
|
|
|
rect->width = cfg->xsize;
|
|
|
|
rect->height = cfg->ysize;
|
|
|
|
break;
|
|
|
|
case V4L2_SEL_TGT_CROP_DEFAULT:
|
|
|
|
case V4L2_SEL_TGT_CROP_BOUNDS:
|
|
|
|
rect->left = 0;
|
|
|
|
rect->top = 0;
|
|
|
|
rect->width = vpbe_dev->current_timings.xres;
|
|
|
|
rect->height = vpbe_dev->current_timings.yres;
|
|
|
|
break;
|
|
|
|
default:
|
2012-12-03 01:50:47 +00:00
|
|
|
return -EINVAL;
|
2011-06-17 07:01:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-10-04 21:38:15 +00:00
|
|
|
static int vpbe_display_g_pixelaspect(struct file *file, void *priv,
|
|
|
|
int type, struct v4l2_fract *f)
|
2011-06-17 07:01:31 +00:00
|
|
|
{
|
2014-10-12 20:40:37 +00:00
|
|
|
struct vpbe_layer *layer = video_drvdata(file);
|
|
|
|
struct vpbe_device *vpbe_dev = layer->disp_dev->vpbe_dev;
|
2011-06-17 07:01:31 +00:00
|
|
|
|
|
|
|
v4l2_dbg(1, debug, &vpbe_dev->v4l2_dev, "VIDIOC_CROPCAP ioctl\n");
|
|
|
|
|
2018-10-04 21:38:15 +00:00
|
|
|
if (type != V4L2_BUF_TYPE_VIDEO_OUTPUT)
|
2016-07-03 10:53:31 +00:00
|
|
|
return -EINVAL;
|
|
|
|
|
2018-10-04 21:38:15 +00:00
|
|
|
*f = vpbe_dev->current_timings.aspect;
|
2011-06-17 07:01:31 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vpbe_display_g_fmt(struct file *file, void *priv,
|
|
|
|
struct v4l2_format *fmt)
|
|
|
|
{
|
2014-10-12 20:40:37 +00:00
|
|
|
struct vpbe_layer *layer = video_drvdata(file);
|
|
|
|
struct vpbe_device *vpbe_dev = layer->disp_dev->vpbe_dev;
|
2011-06-17 07:01:31 +00:00
|
|
|
|
|
|
|
v4l2_dbg(1, debug, &vpbe_dev->v4l2_dev,
|
|
|
|
"VIDIOC_G_FMT, layer id = %d\n",
|
|
|
|
layer->device_id);
|
|
|
|
|
|
|
|
/* If buffer type is video output */
|
|
|
|
if (V4L2_BUF_TYPE_VIDEO_OUTPUT != fmt->type) {
|
|
|
|
v4l2_err(&vpbe_dev->v4l2_dev, "invalid type\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
/* Fill in the information about format */
|
|
|
|
fmt->fmt.pix = layer->pix_fmt;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vpbe_display_enum_fmt(struct file *file, void *priv,
|
|
|
|
struct v4l2_fmtdesc *fmt)
|
|
|
|
{
|
2014-10-12 20:40:37 +00:00
|
|
|
struct vpbe_layer *layer = video_drvdata(file);
|
|
|
|
struct vpbe_device *vpbe_dev = layer->disp_dev->vpbe_dev;
|
2011-06-17 07:01:31 +00:00
|
|
|
unsigned int index = 0;
|
|
|
|
|
|
|
|
v4l2_dbg(1, debug, &vpbe_dev->v4l2_dev,
|
|
|
|
"VIDIOC_ENUM_FMT, layer id = %d\n",
|
|
|
|
layer->device_id);
|
|
|
|
if (fmt->index > 1) {
|
|
|
|
v4l2_err(&vpbe_dev->v4l2_dev, "Invalid format index\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Fill in the information about format */
|
|
|
|
index = fmt->index;
|
|
|
|
memset(fmt, 0, sizeof(*fmt));
|
|
|
|
fmt->index = index;
|
|
|
|
fmt->type = V4L2_BUF_TYPE_VIDEO_OUTPUT;
|
|
|
|
if (index == 0) {
|
2018-09-10 20:20:42 +00:00
|
|
|
strscpy(fmt->description, "YUV 4:2:2 - UYVY",
|
|
|
|
sizeof(fmt->description));
|
2011-06-17 07:01:31 +00:00
|
|
|
fmt->pixelformat = V4L2_PIX_FMT_UYVY;
|
|
|
|
} else {
|
2018-09-10 20:20:42 +00:00
|
|
|
strscpy(fmt->description, "Y/CbCr 4:2:0",
|
|
|
|
sizeof(fmt->description));
|
2011-06-17 07:01:31 +00:00
|
|
|
fmt->pixelformat = V4L2_PIX_FMT_NV12;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vpbe_display_s_fmt(struct file *file, void *priv,
|
|
|
|
struct v4l2_format *fmt)
|
|
|
|
{
|
2014-10-12 20:40:37 +00:00
|
|
|
struct vpbe_layer *layer = video_drvdata(file);
|
|
|
|
struct vpbe_display *disp_dev = layer->disp_dev;
|
2011-06-17 07:01:31 +00:00
|
|
|
struct vpbe_device *vpbe_dev = disp_dev->vpbe_dev;
|
|
|
|
struct osd_layer_config *cfg = &layer->layer_info.config;
|
|
|
|
struct v4l2_pix_format *pixfmt = &fmt->fmt.pix;
|
|
|
|
struct osd_state *osd_device = disp_dev->osd_device;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
v4l2_dbg(1, debug, &vpbe_dev->v4l2_dev,
|
|
|
|
"VIDIOC_S_FMT, layer id = %d\n",
|
|
|
|
layer->device_id);
|
|
|
|
|
2014-10-12 20:40:42 +00:00
|
|
|
if (vb2_is_busy(&layer->buffer_queue))
|
2011-06-17 07:01:31 +00:00
|
|
|
return -EBUSY;
|
2014-10-12 20:40:42 +00:00
|
|
|
|
2011-06-17 07:01:31 +00:00
|
|
|
if (V4L2_BUF_TYPE_VIDEO_OUTPUT != fmt->type) {
|
|
|
|
v4l2_dbg(1, debug, &vpbe_dev->v4l2_dev, "invalid type\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
/* Check for valid pixel format */
|
|
|
|
ret = vpbe_try_format(disp_dev, pixfmt, 1);
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
/* YUV420 is requested, check availability of the
|
|
|
|
other video window */
|
|
|
|
|
|
|
|
layer->pix_fmt = *pixfmt;
|
2013-05-07 04:07:25 +00:00
|
|
|
if (pixfmt->pixelformat == V4L2_PIX_FMT_NV12) {
|
|
|
|
struct vpbe_layer *otherlayer;
|
|
|
|
|
|
|
|
otherlayer = _vpbe_display_get_other_win_layer(disp_dev, layer);
|
|
|
|
/* if other layer is available, only
|
|
|
|
* claim it, do not configure it
|
|
|
|
*/
|
|
|
|
ret = osd_device->ops.request_layer(osd_device,
|
|
|
|
otherlayer->layer_info.id);
|
|
|
|
if (ret < 0) {
|
|
|
|
v4l2_err(&vpbe_dev->v4l2_dev,
|
|
|
|
"Display Manager failed to allocate layer\n");
|
|
|
|
return -EBUSY;
|
|
|
|
}
|
|
|
|
}
|
2011-06-17 07:01:31 +00:00
|
|
|
|
|
|
|
/* Get osd layer config */
|
|
|
|
osd_device->ops.get_layer_config(osd_device,
|
|
|
|
layer->layer_info.id, cfg);
|
|
|
|
/* Store the pixel format in the layer object */
|
|
|
|
cfg->xsize = pixfmt->width;
|
|
|
|
cfg->ysize = pixfmt->height;
|
|
|
|
cfg->line_length = pixfmt->bytesperline;
|
|
|
|
cfg->ypos = 0;
|
|
|
|
cfg->xpos = 0;
|
|
|
|
cfg->interlaced = vpbe_dev->current_timings.interlaced;
|
|
|
|
|
|
|
|
if (V4L2_PIX_FMT_UYVY == pixfmt->pixelformat)
|
2013-05-03 11:39:25 +00:00
|
|
|
cfg->pixfmt = PIXFMT_YCBCRI;
|
2011-06-17 07:01:31 +00:00
|
|
|
|
|
|
|
/* Change of the default pixel format for both video windows */
|
|
|
|
if (V4L2_PIX_FMT_NV12 == pixfmt->pixelformat) {
|
|
|
|
struct vpbe_layer *otherlayer;
|
|
|
|
cfg->pixfmt = PIXFMT_NV12;
|
|
|
|
otherlayer = _vpbe_display_get_other_win_layer(disp_dev,
|
|
|
|
layer);
|
|
|
|
otherlayer->layer_info.config.pixfmt = PIXFMT_NV12;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set the layer config in the osd window */
|
|
|
|
ret = osd_device->ops.set_layer_config(osd_device,
|
|
|
|
layer->layer_info.id, cfg);
|
|
|
|
if (ret < 0) {
|
|
|
|
v4l2_err(&vpbe_dev->v4l2_dev,
|
|
|
|
"Error in S_FMT params:\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Readback and fill the local copy of current pix format */
|
|
|
|
osd_device->ops.get_layer_config(osd_device,
|
|
|
|
layer->layer_info.id, cfg);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vpbe_display_try_fmt(struct file *file, void *priv,
|
|
|
|
struct v4l2_format *fmt)
|
|
|
|
{
|
2014-10-12 20:40:37 +00:00
|
|
|
struct vpbe_layer *layer = video_drvdata(file);
|
|
|
|
struct vpbe_display *disp_dev = layer->disp_dev;
|
|
|
|
struct vpbe_device *vpbe_dev = layer->disp_dev->vpbe_dev;
|
2011-06-17 07:01:31 +00:00
|
|
|
struct v4l2_pix_format *pixfmt = &fmt->fmt.pix;
|
|
|
|
|
|
|
|
v4l2_dbg(1, debug, &vpbe_dev->v4l2_dev, "VIDIOC_TRY_FMT\n");
|
|
|
|
|
|
|
|
if (V4L2_BUF_TYPE_VIDEO_OUTPUT != fmt->type) {
|
|
|
|
v4l2_err(&vpbe_dev->v4l2_dev, "invalid type\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check for valid field format */
|
|
|
|
return vpbe_try_format(disp_dev, pixfmt, 0);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2018-04-05 16:51:00 +00:00
|
|
|
/*
|
2011-06-17 07:01:31 +00:00
|
|
|
* vpbe_display_s_std - Set the given standard in the encoder
|
|
|
|
*
|
|
|
|
* Sets the standard if supported by the current encoder. Return the status.
|
|
|
|
* 0 - success & -EINVAL on error
|
|
|
|
*/
|
|
|
|
static int vpbe_display_s_std(struct file *file, void *priv,
|
2013-03-15 09:10:40 +00:00
|
|
|
v4l2_std_id std_id)
|
2011-06-17 07:01:31 +00:00
|
|
|
{
|
2014-10-12 20:40:37 +00:00
|
|
|
struct vpbe_layer *layer = video_drvdata(file);
|
|
|
|
struct vpbe_device *vpbe_dev = layer->disp_dev->vpbe_dev;
|
2011-06-17 07:01:31 +00:00
|
|
|
int ret;
|
|
|
|
|
|
|
|
v4l2_dbg(1, debug, &vpbe_dev->v4l2_dev, "VIDIOC_S_STD\n");
|
|
|
|
|
2014-10-12 20:40:42 +00:00
|
|
|
if (vb2_is_busy(&layer->buffer_queue))
|
2011-06-17 07:01:31 +00:00
|
|
|
return -EBUSY;
|
2014-10-12 20:40:42 +00:00
|
|
|
|
2017-09-07 20:37:16 +00:00
|
|
|
if (vpbe_dev->ops.s_std) {
|
2011-06-17 07:01:31 +00:00
|
|
|
ret = vpbe_dev->ops.s_std(vpbe_dev, std_id);
|
|
|
|
if (ret) {
|
|
|
|
v4l2_err(&vpbe_dev->v4l2_dev,
|
|
|
|
"Failed to set standard for sub devices\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-04-05 16:51:00 +00:00
|
|
|
/*
|
2011-06-17 07:01:31 +00:00
|
|
|
* vpbe_display_g_std - Get the standard in the current encoder
|
|
|
|
*
|
|
|
|
* Get the standard in the current encoder. Return the status. 0 - success
|
|
|
|
* -EINVAL on error
|
|
|
|
*/
|
|
|
|
static int vpbe_display_g_std(struct file *file, void *priv,
|
|
|
|
v4l2_std_id *std_id)
|
|
|
|
{
|
2014-10-12 20:40:37 +00:00
|
|
|
struct vpbe_layer *layer = video_drvdata(file);
|
|
|
|
struct vpbe_device *vpbe_dev = layer->disp_dev->vpbe_dev;
|
2011-06-17 07:01:31 +00:00
|
|
|
|
|
|
|
v4l2_dbg(1, debug, &vpbe_dev->v4l2_dev, "VIDIOC_G_STD\n");
|
|
|
|
|
|
|
|
/* Get the standard from the current encoder */
|
|
|
|
if (vpbe_dev->current_timings.timings_type & VPBE_ENC_STD) {
|
2012-10-01 14:39:46 +00:00
|
|
|
*std_id = vpbe_dev->current_timings.std_id;
|
2011-06-17 07:01:31 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2018-04-05 16:51:00 +00:00
|
|
|
/*
|
2011-06-17 07:01:31 +00:00
|
|
|
* vpbe_display_enum_output - enumerate outputs
|
|
|
|
*
|
|
|
|
* Enumerates the outputs available at the vpbe display
|
|
|
|
* returns the status, -EINVAL if end of output list
|
|
|
|
*/
|
|
|
|
static int vpbe_display_enum_output(struct file *file, void *priv,
|
|
|
|
struct v4l2_output *output)
|
|
|
|
{
|
2014-10-12 20:40:37 +00:00
|
|
|
struct vpbe_layer *layer = video_drvdata(file);
|
|
|
|
struct vpbe_device *vpbe_dev = layer->disp_dev->vpbe_dev;
|
2011-06-17 07:01:31 +00:00
|
|
|
int ret;
|
|
|
|
|
|
|
|
v4l2_dbg(1, debug, &vpbe_dev->v4l2_dev, "VIDIOC_ENUM_OUTPUT\n");
|
|
|
|
|
|
|
|
/* Enumerate outputs */
|
2017-09-07 20:37:16 +00:00
|
|
|
if (!vpbe_dev->ops.enum_outputs)
|
2011-06-17 07:01:31 +00:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
ret = vpbe_dev->ops.enum_outputs(vpbe_dev, output);
|
|
|
|
if (ret) {
|
|
|
|
v4l2_dbg(1, debug, &vpbe_dev->v4l2_dev,
|
|
|
|
"Failed to enumerate outputs\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-04-05 16:51:00 +00:00
|
|
|
/*
|
2011-06-17 07:01:31 +00:00
|
|
|
* vpbe_display_s_output - Set output to
|
|
|
|
* the output specified by the index
|
|
|
|
*/
|
|
|
|
static int vpbe_display_s_output(struct file *file, void *priv,
|
|
|
|
unsigned int i)
|
|
|
|
{
|
2014-10-12 20:40:37 +00:00
|
|
|
struct vpbe_layer *layer = video_drvdata(file);
|
|
|
|
struct vpbe_device *vpbe_dev = layer->disp_dev->vpbe_dev;
|
2011-06-17 07:01:31 +00:00
|
|
|
int ret;
|
|
|
|
|
|
|
|
v4l2_dbg(1, debug, &vpbe_dev->v4l2_dev, "VIDIOC_S_OUTPUT\n");
|
2014-10-12 20:40:42 +00:00
|
|
|
|
|
|
|
if (vb2_is_busy(&layer->buffer_queue))
|
2011-06-17 07:01:31 +00:00
|
|
|
return -EBUSY;
|
2014-10-12 20:40:42 +00:00
|
|
|
|
2017-09-07 20:37:16 +00:00
|
|
|
if (!vpbe_dev->ops.set_output)
|
2011-06-17 07:01:31 +00:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
ret = vpbe_dev->ops.set_output(vpbe_dev, i);
|
|
|
|
if (ret) {
|
|
|
|
v4l2_err(&vpbe_dev->v4l2_dev,
|
|
|
|
"Failed to set output for sub devices\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-04-05 16:51:00 +00:00
|
|
|
/*
|
2011-06-17 07:01:31 +00:00
|
|
|
* vpbe_display_g_output - Get output from subdevice
|
|
|
|
* for a given by the index
|
|
|
|
*/
|
|
|
|
static int vpbe_display_g_output(struct file *file, void *priv,
|
|
|
|
unsigned int *i)
|
|
|
|
{
|
2014-10-12 20:40:37 +00:00
|
|
|
struct vpbe_layer *layer = video_drvdata(file);
|
|
|
|
struct vpbe_device *vpbe_dev = layer->disp_dev->vpbe_dev;
|
2011-06-17 07:01:31 +00:00
|
|
|
|
|
|
|
v4l2_dbg(1, debug, &vpbe_dev->v4l2_dev, "VIDIOC_G_OUTPUT\n");
|
|
|
|
/* Get the standard from the current encoder */
|
|
|
|
*i = vpbe_dev->current_out_index;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-04-05 16:51:00 +00:00
|
|
|
/*
|
2012-10-01 14:39:46 +00:00
|
|
|
* vpbe_display_enum_dv_timings - Enumerate the dv timings
|
2011-06-17 07:01:31 +00:00
|
|
|
*
|
2012-10-01 14:39:46 +00:00
|
|
|
* enum the timings in the current encoder. Return the status. 0 - success
|
2011-06-17 07:01:31 +00:00
|
|
|
* -EINVAL on error
|
|
|
|
*/
|
|
|
|
static int
|
2012-10-01 14:39:46 +00:00
|
|
|
vpbe_display_enum_dv_timings(struct file *file, void *priv,
|
|
|
|
struct v4l2_enum_dv_timings *timings)
|
2011-06-17 07:01:31 +00:00
|
|
|
{
|
2014-10-12 20:40:37 +00:00
|
|
|
struct vpbe_layer *layer = video_drvdata(file);
|
|
|
|
struct vpbe_device *vpbe_dev = layer->disp_dev->vpbe_dev;
|
2011-06-17 07:01:31 +00:00
|
|
|
int ret;
|
|
|
|
|
2012-10-01 14:39:46 +00:00
|
|
|
v4l2_dbg(1, debug, &vpbe_dev->v4l2_dev, "VIDIOC_ENUM_DV_TIMINGS\n");
|
2011-06-17 07:01:31 +00:00
|
|
|
|
|
|
|
/* Enumerate outputs */
|
2017-09-07 20:37:16 +00:00
|
|
|
if (!vpbe_dev->ops.enum_dv_timings)
|
2011-06-17 07:01:31 +00:00
|
|
|
return -EINVAL;
|
|
|
|
|
2012-10-01 14:39:46 +00:00
|
|
|
ret = vpbe_dev->ops.enum_dv_timings(vpbe_dev, timings);
|
2011-06-17 07:01:31 +00:00
|
|
|
if (ret) {
|
|
|
|
v4l2_err(&vpbe_dev->v4l2_dev,
|
2012-10-01 14:39:46 +00:00
|
|
|
"Failed to enumerate dv timings info\n");
|
2011-06-17 07:01:31 +00:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-04-05 16:51:00 +00:00
|
|
|
/*
|
2012-10-01 14:39:46 +00:00
|
|
|
* vpbe_display_s_dv_timings - Set the dv timings
|
2011-06-17 07:01:31 +00:00
|
|
|
*
|
2012-10-01 14:39:46 +00:00
|
|
|
* Set the timings in the current encoder. Return the status. 0 - success
|
2011-06-17 07:01:31 +00:00
|
|
|
* -EINVAL on error
|
|
|
|
*/
|
|
|
|
static int
|
2012-10-01 14:39:46 +00:00
|
|
|
vpbe_display_s_dv_timings(struct file *file, void *priv,
|
|
|
|
struct v4l2_dv_timings *timings)
|
2011-06-17 07:01:31 +00:00
|
|
|
{
|
2014-10-12 20:40:37 +00:00
|
|
|
struct vpbe_layer *layer = video_drvdata(file);
|
|
|
|
struct vpbe_device *vpbe_dev = layer->disp_dev->vpbe_dev;
|
2011-06-17 07:01:31 +00:00
|
|
|
int ret;
|
|
|
|
|
2012-10-01 14:39:46 +00:00
|
|
|
v4l2_dbg(1, debug, &vpbe_dev->v4l2_dev, "VIDIOC_S_DV_TIMINGS\n");
|
2011-06-17 07:01:31 +00:00
|
|
|
|
2014-10-12 20:40:42 +00:00
|
|
|
if (vb2_is_busy(&layer->buffer_queue))
|
2011-06-17 07:01:31 +00:00
|
|
|
return -EBUSY;
|
|
|
|
|
|
|
|
/* Set the given standard in the encoder */
|
2012-10-01 14:39:46 +00:00
|
|
|
if (!vpbe_dev->ops.s_dv_timings)
|
2011-06-17 07:01:31 +00:00
|
|
|
return -EINVAL;
|
|
|
|
|
2012-10-01 14:39:46 +00:00
|
|
|
ret = vpbe_dev->ops.s_dv_timings(vpbe_dev, timings);
|
2011-06-17 07:01:31 +00:00
|
|
|
if (ret) {
|
|
|
|
v4l2_err(&vpbe_dev->v4l2_dev,
|
2012-10-01 14:39:46 +00:00
|
|
|
"Failed to set the dv timings info\n");
|
2011-06-17 07:01:31 +00:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-04-05 16:51:00 +00:00
|
|
|
/*
|
2012-10-01 14:39:46 +00:00
|
|
|
* vpbe_display_g_dv_timings - Set the dv timings
|
2011-06-17 07:01:31 +00:00
|
|
|
*
|
2012-10-01 14:39:46 +00:00
|
|
|
* Get the timings in the current encoder. Return the status. 0 - success
|
2011-06-17 07:01:31 +00:00
|
|
|
* -EINVAL on error
|
|
|
|
*/
|
|
|
|
static int
|
2012-10-01 14:39:46 +00:00
|
|
|
vpbe_display_g_dv_timings(struct file *file, void *priv,
|
|
|
|
struct v4l2_dv_timings *dv_timings)
|
2011-06-17 07:01:31 +00:00
|
|
|
{
|
2014-10-12 20:40:37 +00:00
|
|
|
struct vpbe_layer *layer = video_drvdata(file);
|
|
|
|
struct vpbe_device *vpbe_dev = layer->disp_dev->vpbe_dev;
|
2011-06-17 07:01:31 +00:00
|
|
|
|
2012-10-01 14:39:46 +00:00
|
|
|
v4l2_dbg(1, debug, &vpbe_dev->v4l2_dev, "VIDIOC_G_DV_TIMINGS\n");
|
2011-06-17 07:01:31 +00:00
|
|
|
|
|
|
|
/* Get the given standard in the encoder */
|
|
|
|
|
|
|
|
if (vpbe_dev->current_timings.timings_type &
|
2013-02-15 18:06:28 +00:00
|
|
|
VPBE_ENC_DV_TIMINGS) {
|
2012-10-01 14:39:46 +00:00
|
|
|
*dv_timings = vpbe_dev->current_timings.dv_timings;
|
2011-06-17 07:01:31 +00:00
|
|
|
} else {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* vpbe_display_open()
|
|
|
|
* It creates object of file handle structure and stores it in private_data
|
|
|
|
* member of filepointer
|
|
|
|
*/
|
|
|
|
static int vpbe_display_open(struct file *file)
|
|
|
|
{
|
|
|
|
struct vpbe_layer *layer = video_drvdata(file);
|
|
|
|
struct vpbe_display *disp_dev = layer->disp_dev;
|
|
|
|
struct vpbe_device *vpbe_dev = disp_dev->vpbe_dev;
|
|
|
|
struct osd_state *osd_device = disp_dev->osd_device;
|
|
|
|
int err;
|
|
|
|
|
2014-10-12 20:40:37 +00:00
|
|
|
/* creating context for file descriptor */
|
|
|
|
err = v4l2_fh_open(file);
|
|
|
|
if (err) {
|
|
|
|
v4l2_err(&vpbe_dev->v4l2_dev, "v4l2_fh_open failed\n");
|
|
|
|
return err;
|
2011-06-17 07:01:31 +00:00
|
|
|
}
|
|
|
|
|
2014-10-12 20:40:37 +00:00
|
|
|
/* leaving if layer is already initialized */
|
|
|
|
if (!v4l2_fh_is_singular_file(file))
|
|
|
|
return err;
|
2011-06-17 07:01:31 +00:00
|
|
|
|
|
|
|
if (!layer->usrs) {
|
2012-06-24 09:16:44 +00:00
|
|
|
if (mutex_lock_interruptible(&layer->opslock))
|
|
|
|
return -ERESTARTSYS;
|
2011-06-17 07:01:31 +00:00
|
|
|
/* First claim the layer for this device */
|
|
|
|
err = osd_device->ops.request_layer(osd_device,
|
|
|
|
layer->layer_info.id);
|
2012-06-24 09:16:44 +00:00
|
|
|
mutex_unlock(&layer->opslock);
|
2011-06-17 07:01:31 +00:00
|
|
|
if (err < 0) {
|
|
|
|
/* Couldn't get layer */
|
|
|
|
v4l2_err(&vpbe_dev->v4l2_dev,
|
|
|
|
"Display Manager failed to allocate layer\n");
|
2014-10-12 20:40:37 +00:00
|
|
|
v4l2_fh_release(file);
|
2011-06-17 07:01:31 +00:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* Increment layer usrs counter */
|
|
|
|
layer->usrs++;
|
|
|
|
v4l2_dbg(1, debug, &vpbe_dev->v4l2_dev,
|
|
|
|
"vpbe display device opened successfully\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* vpbe_display_release()
|
|
|
|
* This function deletes buffer queue, frees the buffers and the davinci
|
|
|
|
* display file * handle
|
|
|
|
*/
|
|
|
|
static int vpbe_display_release(struct file *file)
|
|
|
|
{
|
2014-10-12 20:40:37 +00:00
|
|
|
struct vpbe_layer *layer = video_drvdata(file);
|
2011-06-17 07:01:31 +00:00
|
|
|
struct osd_layer_config *cfg = &layer->layer_info.config;
|
2014-10-12 20:40:37 +00:00
|
|
|
struct vpbe_display *disp_dev = layer->disp_dev;
|
2011-06-17 07:01:31 +00:00
|
|
|
struct vpbe_device *vpbe_dev = disp_dev->vpbe_dev;
|
|
|
|
struct osd_state *osd_device = disp_dev->osd_device;
|
|
|
|
|
|
|
|
v4l2_dbg(1, debug, &vpbe_dev->v4l2_dev, "vpbe_display_release\n");
|
|
|
|
|
2012-06-24 09:16:44 +00:00
|
|
|
mutex_lock(&layer->opslock);
|
2011-06-17 07:01:31 +00:00
|
|
|
|
2014-10-12 20:40:37 +00:00
|
|
|
osd_device->ops.disable_layer(osd_device,
|
|
|
|
layer->layer_info.id);
|
2011-06-17 07:01:31 +00:00
|
|
|
/* Decrement layer usrs counter */
|
|
|
|
layer->usrs--;
|
|
|
|
/* If this file handle has initialize encoder device, reset it */
|
|
|
|
if (!layer->usrs) {
|
|
|
|
if (cfg->pixfmt == PIXFMT_NV12) {
|
|
|
|
struct vpbe_layer *otherlayer;
|
|
|
|
otherlayer =
|
|
|
|
_vpbe_display_get_other_win_layer(disp_dev, layer);
|
|
|
|
osd_device->ops.disable_layer(osd_device,
|
|
|
|
otherlayer->layer_info.id);
|
|
|
|
osd_device->ops.release_layer(osd_device,
|
|
|
|
otherlayer->layer_info.id);
|
|
|
|
}
|
|
|
|
osd_device->ops.disable_layer(osd_device,
|
|
|
|
layer->layer_info.id);
|
|
|
|
osd_device->ops.release_layer(osd_device,
|
|
|
|
layer->layer_info.id);
|
|
|
|
}
|
2014-03-22 10:57:59 +00:00
|
|
|
|
2014-10-12 20:40:37 +00:00
|
|
|
_vb2_fop_release(file, NULL);
|
2012-06-24 09:16:44 +00:00
|
|
|
mutex_unlock(&layer->opslock);
|
2011-06-17 07:01:31 +00:00
|
|
|
|
|
|
|
disp_dev->cbcr_ofst = 0;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* vpbe capture ioctl operations */
|
|
|
|
static const struct v4l2_ioctl_ops vpbe_ioctl_ops = {
|
|
|
|
.vidioc_querycap = vpbe_display_querycap,
|
|
|
|
.vidioc_g_fmt_vid_out = vpbe_display_g_fmt,
|
|
|
|
.vidioc_enum_fmt_vid_out = vpbe_display_enum_fmt,
|
|
|
|
.vidioc_s_fmt_vid_out = vpbe_display_s_fmt,
|
|
|
|
.vidioc_try_fmt_vid_out = vpbe_display_try_fmt,
|
2014-10-12 20:40:38 +00:00
|
|
|
|
|
|
|
.vidioc_reqbufs = vb2_ioctl_reqbufs,
|
2014-10-22 21:42:01 +00:00
|
|
|
.vidioc_create_bufs = vb2_ioctl_create_bufs,
|
2014-10-12 20:40:38 +00:00
|
|
|
.vidioc_querybuf = vb2_ioctl_querybuf,
|
|
|
|
.vidioc_qbuf = vb2_ioctl_qbuf,
|
|
|
|
.vidioc_dqbuf = vb2_ioctl_dqbuf,
|
|
|
|
.vidioc_streamon = vb2_ioctl_streamon,
|
|
|
|
.vidioc_streamoff = vb2_ioctl_streamoff,
|
2014-10-12 20:40:41 +00:00
|
|
|
.vidioc_expbuf = vb2_ioctl_expbuf,
|
2014-10-12 20:40:38 +00:00
|
|
|
|
2018-10-04 21:38:15 +00:00
|
|
|
.vidioc_g_pixelaspect = vpbe_display_g_pixelaspect,
|
2016-07-03 10:53:31 +00:00
|
|
|
.vidioc_g_selection = vpbe_display_g_selection,
|
|
|
|
.vidioc_s_selection = vpbe_display_s_selection,
|
2014-10-12 20:40:44 +00:00
|
|
|
|
2011-06-17 07:01:31 +00:00
|
|
|
.vidioc_s_std = vpbe_display_s_std,
|
|
|
|
.vidioc_g_std = vpbe_display_g_std,
|
2014-10-12 20:40:44 +00:00
|
|
|
|
2011-06-17 07:01:31 +00:00
|
|
|
.vidioc_enum_output = vpbe_display_enum_output,
|
|
|
|
.vidioc_s_output = vpbe_display_s_output,
|
|
|
|
.vidioc_g_output = vpbe_display_g_output,
|
2014-10-12 20:40:44 +00:00
|
|
|
|
2012-10-01 14:39:46 +00:00
|
|
|
.vidioc_s_dv_timings = vpbe_display_s_dv_timings,
|
|
|
|
.vidioc_g_dv_timings = vpbe_display_g_dv_timings,
|
|
|
|
.vidioc_enum_dv_timings = vpbe_display_enum_dv_timings,
|
2011-06-17 07:01:31 +00:00
|
|
|
};
|
|
|
|
|
2017-06-29 08:51:24 +00:00
|
|
|
static const struct v4l2_file_operations vpbe_fops = {
|
2011-06-17 07:01:31 +00:00
|
|
|
.owner = THIS_MODULE,
|
|
|
|
.open = vpbe_display_open,
|
|
|
|
.release = vpbe_display_release,
|
|
|
|
.unlocked_ioctl = video_ioctl2,
|
2014-10-12 20:40:36 +00:00
|
|
|
.mmap = vb2_fop_mmap,
|
|
|
|
.poll = vb2_fop_poll,
|
2011-06-17 07:01:31 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static int vpbe_device_get(struct device *dev, void *data)
|
|
|
|
{
|
|
|
|
struct platform_device *pdev = to_platform_device(dev);
|
|
|
|
struct vpbe_display *vpbe_disp = data;
|
|
|
|
|
|
|
|
if (strcmp("vpbe_controller", pdev->name) == 0)
|
|
|
|
vpbe_disp->vpbe_dev = platform_get_drvdata(pdev);
|
|
|
|
|
2017-09-07 20:37:16 +00:00
|
|
|
if (strstr(pdev->name, "vpbe-osd"))
|
2011-06-17 07:01:31 +00:00
|
|
|
vpbe_disp->osd_device = platform_get_drvdata(pdev);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-12-21 21:17:53 +00:00
|
|
|
static int init_vpbe_layer(int i, struct vpbe_display *disp_dev,
|
|
|
|
struct platform_device *pdev)
|
2011-06-17 07:01:31 +00:00
|
|
|
{
|
|
|
|
struct vpbe_layer *vpbe_display_layer = NULL;
|
|
|
|
struct video_device *vbd = NULL;
|
|
|
|
|
|
|
|
/* Allocate memory for four plane display objects */
|
2017-09-07 20:37:16 +00:00
|
|
|
disp_dev->dev[i] = kzalloc(sizeof(*disp_dev->dev[i]), GFP_KERNEL);
|
|
|
|
if (!disp_dev->dev[i])
|
2011-06-17 07:01:31 +00:00
|
|
|
return -ENOMEM;
|
2017-09-07 20:37:16 +00:00
|
|
|
|
2011-06-17 07:01:31 +00:00
|
|
|
spin_lock_init(&disp_dev->dev[i]->irqlock);
|
|
|
|
mutex_init(&disp_dev->dev[i]->opslock);
|
|
|
|
|
|
|
|
/* Get the pointer to the layer object */
|
|
|
|
vpbe_display_layer = disp_dev->dev[i];
|
|
|
|
vbd = &vpbe_display_layer->video_dev;
|
|
|
|
/* Initialize field of video device */
|
|
|
|
vbd->release = video_device_release_empty;
|
|
|
|
vbd->fops = &vpbe_fops;
|
|
|
|
vbd->ioctl_ops = &vpbe_ioctl_ops;
|
|
|
|
vbd->minor = -1;
|
|
|
|
vbd->v4l2_dev = &disp_dev->vpbe_dev->v4l2_dev;
|
|
|
|
vbd->lock = &vpbe_display_layer->opslock;
|
2012-09-05 09:05:50 +00:00
|
|
|
vbd->vfl_dir = VFL_DIR_TX;
|
2011-06-17 07:01:31 +00:00
|
|
|
|
|
|
|
if (disp_dev->vpbe_dev->current_timings.timings_type &
|
2013-02-19 16:33:34 +00:00
|
|
|
VPBE_ENC_STD)
|
2011-06-17 07:01:31 +00:00
|
|
|
vbd->tvnorms = (V4L2_STD_525_60 | V4L2_STD_625_50);
|
|
|
|
|
|
|
|
snprintf(vbd->name, sizeof(vbd->name),
|
|
|
|
"DaVinci_VPBE Display_DRIVER_V%d.%d.%d",
|
|
|
|
(VPBE_DISPLAY_VERSION_CODE >> 16) & 0xff,
|
|
|
|
(VPBE_DISPLAY_VERSION_CODE >> 8) & 0xff,
|
|
|
|
(VPBE_DISPLAY_VERSION_CODE) & 0xff);
|
|
|
|
|
|
|
|
vpbe_display_layer->device_id = i;
|
|
|
|
|
|
|
|
vpbe_display_layer->layer_info.id =
|
|
|
|
((i == VPBE_DISPLAY_DEVICE_0) ? WIN_VID0 : WIN_VID1);
|
|
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-12-21 21:17:53 +00:00
|
|
|
static int register_device(struct vpbe_layer *vpbe_display_layer,
|
|
|
|
struct vpbe_display *disp_dev,
|
|
|
|
struct platform_device *pdev)
|
|
|
|
{
|
2011-06-17 07:01:31 +00:00
|
|
|
int err;
|
|
|
|
|
|
|
|
v4l2_info(&disp_dev->vpbe_dev->v4l2_dev,
|
|
|
|
"Trying to register VPBE display device.\n");
|
|
|
|
v4l2_info(&disp_dev->vpbe_dev->v4l2_dev,
|
2018-04-06 12:24:27 +00:00
|
|
|
"layer=%p,layer->video_dev=%p\n",
|
|
|
|
vpbe_display_layer,
|
|
|
|
&vpbe_display_layer->video_dev);
|
2011-06-17 07:01:31 +00:00
|
|
|
|
2014-10-12 20:40:36 +00:00
|
|
|
vpbe_display_layer->video_dev.queue = &vpbe_display_layer->buffer_queue;
|
2011-06-17 07:01:31 +00:00
|
|
|
err = video_register_device(&vpbe_display_layer->video_dev,
|
|
|
|
VFL_TYPE_GRABBER,
|
|
|
|
-1);
|
|
|
|
if (err)
|
|
|
|
return -ENODEV;
|
|
|
|
|
|
|
|
vpbe_display_layer->disp_dev = disp_dev;
|
|
|
|
/* set the driver data in platform device */
|
|
|
|
platform_set_drvdata(pdev, disp_dev);
|
|
|
|
video_set_drvdata(&vpbe_display_layer->video_dev,
|
|
|
|
vpbe_display_layer);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* vpbe_display_probe()
|
|
|
|
* This function creates device entries by register itself to the V4L2 driver
|
|
|
|
* and initializes fields of each layer objects
|
|
|
|
*/
|
2012-12-21 21:17:53 +00:00
|
|
|
static int vpbe_display_probe(struct platform_device *pdev)
|
2011-06-17 07:01:31 +00:00
|
|
|
{
|
|
|
|
struct vpbe_display *disp_dev;
|
2014-10-12 20:40:31 +00:00
|
|
|
struct v4l2_device *v4l2_dev;
|
2011-06-17 07:01:31 +00:00
|
|
|
struct resource *res = NULL;
|
2014-10-12 20:40:31 +00:00
|
|
|
struct vb2_queue *q;
|
2011-06-17 07:01:31 +00:00
|
|
|
int k;
|
|
|
|
int i;
|
|
|
|
int err;
|
|
|
|
int irq;
|
|
|
|
|
|
|
|
printk(KERN_DEBUG "vpbe_display_probe\n");
|
|
|
|
/* Allocate memory for vpbe_display */
|
2017-09-07 20:37:16 +00:00
|
|
|
disp_dev = devm_kzalloc(&pdev->dev, sizeof(*disp_dev), GFP_KERNEL);
|
2013-07-13 07:50:29 +00:00
|
|
|
if (!disp_dev)
|
2011-06-17 07:01:31 +00:00
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
spin_lock_init(&disp_dev->dma_queue_lock);
|
|
|
|
/*
|
|
|
|
* Scan all the platform devices to find the vpbe
|
|
|
|
* controller device and get the vpbe_dev object
|
|
|
|
*/
|
|
|
|
err = bus_for_each_dev(&platform_bus_type, NULL, disp_dev,
|
|
|
|
vpbe_device_get);
|
|
|
|
if (err < 0)
|
|
|
|
return err;
|
2014-10-12 20:40:31 +00:00
|
|
|
|
|
|
|
v4l2_dev = &disp_dev->vpbe_dev->v4l2_dev;
|
2011-06-17 07:01:31 +00:00
|
|
|
/* Initialize the vpbe display controller */
|
2017-09-07 20:37:16 +00:00
|
|
|
if (disp_dev->vpbe_dev->ops.initialize) {
|
2011-06-17 07:01:31 +00:00
|
|
|
err = disp_dev->vpbe_dev->ops.initialize(&pdev->dev,
|
|
|
|
disp_dev->vpbe_dev);
|
|
|
|
if (err) {
|
2014-10-12 20:40:31 +00:00
|
|
|
v4l2_err(v4l2_dev, "Error initing vpbe\n");
|
2011-06-17 07:01:31 +00:00
|
|
|
err = -ENOMEM;
|
|
|
|
goto probe_out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < VPBE_DISPLAY_MAX_DEVICES; i++) {
|
|
|
|
if (init_vpbe_layer(i, disp_dev, pdev)) {
|
|
|
|
err = -ENODEV;
|
|
|
|
goto probe_out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
res = platform_get_resource(pdev, IORESOURCE_IRQ, 0);
|
|
|
|
if (!res) {
|
2014-10-12 20:40:31 +00:00
|
|
|
v4l2_err(v4l2_dev, "Unable to get VENC interrupt resource\n");
|
2011-06-17 07:01:31 +00:00
|
|
|
err = -ENODEV;
|
|
|
|
goto probe_out;
|
|
|
|
}
|
|
|
|
|
|
|
|
irq = res->start;
|
2013-09-09 02:30:11 +00:00
|
|
|
err = devm_request_irq(&pdev->dev, irq, venc_isr, 0,
|
2013-07-13 07:50:29 +00:00
|
|
|
VPBE_DISPLAY_DRIVER, disp_dev);
|
|
|
|
if (err) {
|
2014-10-12 20:40:31 +00:00
|
|
|
v4l2_err(v4l2_dev, "VPBE IRQ request failed\n");
|
2011-06-17 07:01:31 +00:00
|
|
|
goto probe_out;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < VPBE_DISPLAY_MAX_DEVICES; i++) {
|
2014-10-12 20:40:31 +00:00
|
|
|
/* initialize vb2 queue */
|
|
|
|
q = &disp_dev->dev[i]->buffer_queue;
|
|
|
|
memset(q, 0, sizeof(*q));
|
|
|
|
q->type = V4L2_BUF_TYPE_VIDEO_OUTPUT;
|
2014-10-12 20:40:39 +00:00
|
|
|
q->io_modes = VB2_MMAP | VB2_USERPTR | VB2_DMABUF;
|
2014-10-12 20:40:31 +00:00
|
|
|
q->drv_priv = disp_dev->dev[i];
|
|
|
|
q->ops = &video_qops;
|
|
|
|
q->mem_ops = &vb2_dma_contig_memops;
|
|
|
|
q->buf_struct_size = sizeof(struct vpbe_disp_buffer);
|
|
|
|
q->timestamp_flags = V4L2_BUF_FLAG_TIMESTAMP_MONOTONIC;
|
|
|
|
q->min_buffers_needed = 1;
|
2014-10-12 20:40:33 +00:00
|
|
|
q->lock = &disp_dev->dev[i]->opslock;
|
2016-02-15 15:09:10 +00:00
|
|
|
q->dev = disp_dev->vpbe_dev->pdev;
|
2014-10-12 20:40:31 +00:00
|
|
|
err = vb2_queue_init(q);
|
|
|
|
if (err) {
|
|
|
|
v4l2_err(v4l2_dev, "vb2_queue_init() failed\n");
|
|
|
|
goto probe_out;
|
|
|
|
}
|
|
|
|
|
|
|
|
INIT_LIST_HEAD(&disp_dev->dev[i]->dma_queue);
|
|
|
|
|
2011-06-17 07:01:31 +00:00
|
|
|
if (register_device(disp_dev->dev[i], disp_dev, pdev)) {
|
|
|
|
err = -ENODEV;
|
2013-07-13 07:50:29 +00:00
|
|
|
goto probe_out;
|
2011-06-17 07:01:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-12 20:40:31 +00:00
|
|
|
v4l2_dbg(1, debug, v4l2_dev,
|
|
|
|
"Successfully completed the probing of vpbe v4l2 device\n");
|
|
|
|
|
2011-06-17 07:01:31 +00:00
|
|
|
return 0;
|
|
|
|
|
2011-10-28 22:58:17 +00:00
|
|
|
probe_out:
|
2011-06-17 07:01:31 +00:00
|
|
|
for (k = 0; k < VPBE_DISPLAY_MAX_DEVICES; k++) {
|
|
|
|
/* Unregister video device */
|
2017-09-07 20:37:16 +00:00
|
|
|
if (disp_dev->dev[k]) {
|
2014-10-12 20:40:31 +00:00
|
|
|
video_unregister_device(&disp_dev->dev[k]->video_dev);
|
|
|
|
kfree(disp_dev->dev[k]);
|
2011-06-17 07:01:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* vpbe_display_remove()
|
|
|
|
* It un-register hardware layer from V4L2 driver
|
|
|
|
*/
|
|
|
|
static int vpbe_display_remove(struct platform_device *pdev)
|
|
|
|
{
|
|
|
|
struct vpbe_layer *vpbe_display_layer;
|
|
|
|
struct vpbe_display *disp_dev = platform_get_drvdata(pdev);
|
|
|
|
struct vpbe_device *vpbe_dev = disp_dev->vpbe_dev;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
v4l2_dbg(1, debug, &vpbe_dev->v4l2_dev, "vpbe_display_remove\n");
|
|
|
|
|
|
|
|
/* deinitialize the vpbe display controller */
|
2017-09-07 20:37:16 +00:00
|
|
|
if (vpbe_dev->ops.deinitialize)
|
2011-06-17 07:01:31 +00:00
|
|
|
vpbe_dev->ops.deinitialize(&pdev->dev, vpbe_dev);
|
|
|
|
/* un-register device */
|
|
|
|
for (i = 0; i < VPBE_DISPLAY_MAX_DEVICES; i++) {
|
|
|
|
/* Get the pointer to the layer object */
|
|
|
|
vpbe_display_layer = disp_dev->dev[i];
|
|
|
|
/* Unregister video device */
|
|
|
|
video_unregister_device(&vpbe_display_layer->video_dev);
|
|
|
|
|
|
|
|
}
|
|
|
|
for (i = 0; i < VPBE_DISPLAY_MAX_DEVICES; i++) {
|
|
|
|
kfree(disp_dev->dev[i]);
|
|
|
|
disp_dev->dev[i] = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct platform_driver vpbe_display_driver = {
|
|
|
|
.driver = {
|
|
|
|
.name = VPBE_DISPLAY_DRIVER,
|
|
|
|
.bus = &platform_bus_type,
|
|
|
|
},
|
|
|
|
.probe = vpbe_display_probe,
|
2012-12-21 21:17:53 +00:00
|
|
|
.remove = vpbe_display_remove,
|
2011-06-17 07:01:31 +00:00
|
|
|
};
|
|
|
|
|
2012-01-10 06:21:49 +00:00
|
|
|
module_platform_driver(vpbe_display_driver);
|
2011-06-17 07:01:31 +00:00
|
|
|
|
|
|
|
MODULE_DESCRIPTION("TI DM644x/DM355/DM365 VPBE Display controller");
|
|
|
|
MODULE_LICENSE("GPL");
|
|
|
|
MODULE_AUTHOR("Texas Instruments");
|