lineage_kernel_xcoverpro/drivers/media/platform/exynos/fimc-is2/fimc-is-video-3aa.c

932 lines
23 KiB
C
Executable File

/*
* Samsung Exynos5 SoC series FIMC-IS driver
*
* exynos5 fimc-is video functions
*
* Copyright (c) 2011 Samsung Electronics Co., Ltd
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation.
*/
#include <linux/module.h>
#include <linux/delay.h>
#include <linux/kernel.h>
#include <linux/errno.h>
#include <linux/interrupt.h>
#include <linux/device.h>
#include <linux/platform_device.h>
#include <linux/slab.h>
#include <linux/firmware.h>
#include <linux/dma-mapping.h>
#include <linux/scatterlist.h>
#include <linux/videodev2_exynos_camera.h>
#include <linux/v4l2-mediabus.h>
#include <linux/bug.h>
#include <media/videobuf2-v4l2.h>
#include <media/v4l2-ctrls.h>
#include <media/v4l2-device.h>
#include <media/v4l2-mem2mem.h>
#include <media/v4l2-mediabus.h>
#include "fimc-is-core.h"
#include "fimc-is-cmd.h"
#include "fimc-is-regs.h"
#include "fimc-is-err.h"
#include "fimc-is-video.h"
#include "fimc-is-param.h"
const struct v4l2_file_operations fimc_is_3aa_video_fops;
const struct v4l2_ioctl_ops fimc_is_3aa_video_ioctl_ops;
const struct vb2_ops fimc_is_3aa_qops;
int fimc_is_30s_video_probe(void *data)
{
int ret = 0;
struct fimc_is_core *core;
struct fimc_is_video *video;
FIMC_BUG(!data);
core = (struct fimc_is_core *)data;
video = &core->video_30s;
video->resourcemgr = &core->resourcemgr;
if (!core->pdev) {
probe_err("pdev is NULL");
ret = -EINVAL;
goto p_err;
}
ret = fimc_is_video_probe(video,
FIMC_IS_VIDEO_3XS_NAME(0),
FIMC_IS_VIDEO_30S_NUM,
VFL_DIR_M2M,
&core->resourcemgr.mem,
&core->v4l2_dev,
&fimc_is_3aa_video_fops,
&fimc_is_3aa_video_ioctl_ops);
if (ret)
dev_err(&core->pdev->dev, "%s is fail(%d)\n", __func__, ret);
p_err:
return ret;
}
int fimc_is_31s_video_probe(void *data)
{
int ret = 0;
struct fimc_is_core *core;
struct fimc_is_video *video;
FIMC_BUG(!data);
core = (struct fimc_is_core *)data;
video = &core->video_31s;
video->resourcemgr = &core->resourcemgr;
if (!core->pdev) {
probe_err("pdev is NULL");
ret = -EINVAL;
goto p_err;
}
ret = fimc_is_video_probe(video,
FIMC_IS_VIDEO_3XS_NAME(1),
FIMC_IS_VIDEO_31S_NUM,
VFL_DIR_M2M,
&core->resourcemgr.mem,
&core->v4l2_dev,
&fimc_is_3aa_video_fops,
&fimc_is_3aa_video_ioctl_ops);
if (ret)
dev_err(&core->pdev->dev, "%s is fail(%d)\n", __func__, ret);
p_err:
return ret;
}
int fimc_is_32s_video_probe(void *data)
{
int ret = 0;
struct fimc_is_core *core;
struct fimc_is_video *video;
FIMC_BUG(!data);
core = (struct fimc_is_core *)data;
video = &core->video_32s;
video->resourcemgr = &core->resourcemgr;
if (!core->pdev) {
probe_err("pdev is NULL");
ret = -EINVAL;
goto p_err;
}
ret = fimc_is_video_probe(video,
FIMC_IS_VIDEO_3XS_NAME(2),
FIMC_IS_VIDEO_32S_NUM,
VFL_DIR_M2M,
&core->resourcemgr.mem,
&core->v4l2_dev,
&fimc_is_3aa_video_fops,
&fimc_is_3aa_video_ioctl_ops);
if (ret)
dev_err(&core->pdev->dev, "%s is fail(%d)\n", __func__, ret);
p_err:
return ret;
}
/*
* =============================================================================
* Video File Opertation
* =============================================================================
*/
static int fimc_is_3aa_video_open(struct file *file)
{
int ret = 0;
int ret_err = 0;
struct fimc_is_video *video;
struct fimc_is_video_ctx *vctx;
struct fimc_is_device_ischain *device;
struct fimc_is_resourcemgr *resourcemgr;
char name[FIMC_IS_STR_LEN];
vctx = NULL;
device = NULL;
video = video_drvdata(file);
resourcemgr = video->resourcemgr;
if (!resourcemgr) {
err("resourcemgr is NULL");
ret = -EINVAL;
goto err_resource_null;
}
ret = fimc_is_resource_open(resourcemgr, RESOURCE_TYPE_ISCHAIN, (void **)&device);
if (ret) {
err("fimc_is_resource_open is fail(%d)", ret);
goto err_resource_open;
}
if (!device) {
err("device is NULL");
ret = -EINVAL;
goto err_device_null;
}
minfo("[3%dS:V] %s\n", device, GET_3XS_ID(video), __func__);
snprintf(name, sizeof(name), "3%dS", GET_3XS_ID(video));
ret = open_vctx(file, video, &vctx, device->instance, FRAMEMGR_ID_3XS, name);
if (ret) {
merr("open_vctx is fail(%d)", device, ret);
goto err_vctx_open;
}
ret = fimc_is_video_open(vctx,
device,
VIDEO_3XS_READY_BUFFERS,
video,
&fimc_is_3aa_qops,
&fimc_is_ischain_3aa_ops);
if (ret) {
merr("fimc_is_video_open is fail(%d)", device, ret);
goto err_video_open;
}
ret = fimc_is_ischain_3aa_open(device, vctx);
if (ret) {
merr("fimc_is_ischain_3aa_open is fail(%d)", device, ret);
goto err_ischain_open;
}
return 0;
err_ischain_open:
ret_err = fimc_is_video_close(vctx);
if (ret_err)
merr("fimc_is_video_close is fail(%d)", device, ret_err);
err_video_open:
ret_err = close_vctx(file, video, vctx);
if (ret_err < 0)
merr("close_vctx is fail(%d)", device, ret_err);
err_vctx_open:
err_device_null:
err_resource_open:
err_resource_null:
return ret;
}
static int fimc_is_3aa_video_close(struct file *file)
{
int ret = 0;
int refcount;
struct fimc_is_video_ctx *vctx = file->private_data;
struct fimc_is_video *video;
struct fimc_is_device_ischain *device;
FIMC_BUG(!file);
FIMC_BUG(!vctx);
FIMC_BUG(!GET_VIDEO(vctx));
FIMC_BUG(!GET_DEVICE(vctx));
video = GET_VIDEO(vctx);
device = GET_DEVICE(vctx);
ret = fimc_is_ischain_3aa_close(device, vctx);
if (ret)
merr("fimc_is_ischain_3aa_close is fail(%d)", device, ret);
ret = fimc_is_video_close(vctx);
if (ret)
merr("fimc_is_video_close is fail(%d)", device, ret);
refcount = close_vctx(file, video, vctx);
if (refcount < 0)
merr("close_vctx is fail(%d)", device, refcount);
minfo("[3%dS:V] %s(%d,%d):%d\n", device, GET_3XS_ID(video), __func__, atomic_read(&device->open_cnt), refcount, ret);
return ret;
}
static unsigned int fimc_is_3aa_video_poll(struct file *file,
struct poll_table_struct *wait)
{
u32 ret = 0;
struct fimc_is_video_ctx *vctx = file->private_data;
ret = fimc_is_video_poll(file, vctx, wait);
if (ret)
merr("fimc_is_video_poll is fail(%d)", vctx, ret);
return ret;
}
static int fimc_is_3aa_video_mmap(struct file *file,
struct vm_area_struct *vma)
{
int ret = 0;
struct fimc_is_video_ctx *vctx = file->private_data;
ret = fimc_is_video_mmap(file, vctx, vma);
if (ret)
merr("fimc_is_video_mmap is fail(%d)", vctx, ret);
return ret;
}
const struct v4l2_file_operations fimc_is_3aa_video_fops = {
.owner = THIS_MODULE,
.open = fimc_is_3aa_video_open,
.release = fimc_is_3aa_video_close,
.poll = fimc_is_3aa_video_poll,
.unlocked_ioctl = video_ioctl2,
.mmap = fimc_is_3aa_video_mmap,
};
/*
* =============================================================================
* Video Ioctl Opertation
* =============================================================================
*/
static int fimc_is_3aa_video_querycap(struct file *file, void *fh,
struct v4l2_capability *cap)
{
struct fimc_is_video *video = video_drvdata(file);
FIMC_BUG(!cap);
FIMC_BUG(!video);
snprintf(cap->driver, sizeof(cap->driver), "%s", video->vd.name);
snprintf(cap->card, sizeof(cap->card), "%s", video->vd.name);
cap->capabilities |= V4L2_CAP_STREAMING
| V4L2_CAP_VIDEO_OUTPUT
| V4L2_CAP_VIDEO_OUTPUT_MPLANE;
cap->device_caps |= cap->capabilities;
return 0;
}
static int fimc_is_3aa_video_enum_fmt_mplane(struct file *file, void *priv,
struct v4l2_fmtdesc *f)
{
/* Todo : add to enumerate format code */
return 0;
}
static int fimc_is_3aa_video_get_format_mplane(struct file *file, void *fh,
struct v4l2_format *format)
{
/* Todo : add to get format code */
return 0;
}
static int fimc_is_3aa_video_set_format_mplane(struct file *file, void *fh,
struct v4l2_format *format)
{
int ret = 0;
struct fimc_is_video_ctx *vctx = file->private_data;
FIMC_BUG(!vctx);
FIMC_BUG(!format);
mdbgv_3aa("%s\n", vctx, __func__);
ret = fimc_is_video_set_format_mplane(file, vctx, format);
if (ret) {
merr("fimc_is_video_set_format_mplane is fail(%d)", vctx, ret);
goto p_err;
}
p_err:
return ret;
}
static int fimc_is_3aa_video_cropcap(struct file *file, void *fh,
struct v4l2_cropcap *cropcap)
{
/* Todo : add to crop capability code */
return 0;
}
static int fimc_is_3aa_video_get_crop(struct file *file, void *fh,
struct v4l2_crop *crop)
{
/* Todo : add to get crop control code */
return 0;
}
static int fimc_is_3aa_video_set_crop(struct file *file, void *fh,
const struct v4l2_crop *crop)
{
/* Todo : add to set crop control code */
return 0;
}
static int fimc_is_3aa_video_reqbufs(struct file *file, void *priv,
struct v4l2_requestbuffers *buf)
{
int ret = 0;
struct fimc_is_video_ctx *vctx = file->private_data;
FIMC_BUG(!vctx);
mdbgv_3aa("%s(buffers : %d)\n", vctx, __func__, buf->count);
ret = fimc_is_video_reqbufs(file, vctx, buf);
if (ret)
merr("fimc_is_video_reqbufs is fail(%d)", vctx, ret);
return ret;
}
static int fimc_is_3aa_video_querybuf(struct file *file, void *priv,
struct v4l2_buffer *buf)
{
int ret;
struct fimc_is_video_ctx *vctx = file->private_data;
mdbgv_3aa("%s\n", vctx, __func__);
ret = fimc_is_video_querybuf(file, vctx, buf);
if (ret)
merr("fimc_is_video_querybuf is fail(%d)", vctx, ret);
return ret;
}
static int fimc_is_3aa_video_qbuf(struct file *file, void *priv,
struct v4l2_buffer *buf)
{
int ret = 0;
struct fimc_is_video_ctx *vctx = file->private_data;
FIMC_BUG(!vctx);
mvdbgs(3, "%s(%02d:%d)\n", vctx, &vctx->queue, __func__, buf->type, buf->index);
ret = CALL_VOPS(vctx, qbuf, buf);
if (ret)
merr("qbuf is fail(%d)", vctx, ret);
return ret;
}
static int fimc_is_3aa_video_dqbuf(struct file *file, void *priv,
struct v4l2_buffer *buf)
{
int ret = 0;
struct fimc_is_video_ctx *vctx = file->private_data;
bool blocking = file->f_flags & O_NONBLOCK;
FIMC_BUG(!vctx);
mvdbgs(3, "%s\n", vctx, &vctx->queue, __func__);
ret = CALL_VOPS(vctx, dqbuf, buf, blocking);
if (ret)
merr("dqbuf is fail(%d)", vctx, ret);
return ret;
}
static int fimc_is_3aa_video_prepare(struct file *file, void *priv,
struct v4l2_buffer *buf)
{
int ret = 0;
struct fimc_is_video_ctx *vctx = file->private_data;
struct fimc_is_device_ischain *device;
struct fimc_is_framemgr *framemgr;
struct fimc_is_frame *frame;
FIMC_BUG(!buf);
FIMC_BUG(!vctx);
FIMC_BUG(!GET_FRAMEMGR(vctx));
FIMC_BUG(!GET_DEVICE(vctx));
FIMC_BUG(!GET_VIDEO(vctx));
device = GET_DEVICE(vctx);
framemgr = GET_FRAMEMGR(vctx);
frame = &framemgr->frames[buf->index];
ret = fimc_is_video_prepare(file, vctx, buf);
if (ret) {
merr("fimc_is_video_prepare is fail(%d)", vctx, ret);
goto p_err;
}
#ifdef ENABLE_IS_CORE
if (!test_bit(FRAME_MEM_MAPPED, &frame->mem_state)) {
fimc_is_itf_map(device, GROUP_ID(device->group_3aa.id), frame->dvaddr_shot, frame->shot_size);
set_bit(FRAME_MEM_MAPPED, &frame->mem_state);
}
#endif
p_err:
minfo("[3%dS:V] %s(%d):%d\n", device, GET_3XS_ID(GET_VIDEO(vctx)), __func__, buf->index, ret);
return ret;
}
static int fimc_is_3aa_video_streamon(struct file *file, void *priv,
enum v4l2_buf_type type)
{
int ret = 0;
struct fimc_is_video_ctx *vctx = file->private_data;
mdbgv_3aa("%s\n", vctx, __func__);
ret = fimc_is_video_streamon(file, vctx, type);
if (ret)
merr("fimc_is_video_streamon is fail(%d)", vctx, ret);
return ret;
}
static int fimc_is_3aa_video_streamoff(struct file *file, void *priv,
enum v4l2_buf_type type)
{
int ret = 0;
struct fimc_is_video_ctx *vctx = file->private_data;
mdbgv_3aa("%s\n", vctx, __func__);
ret = fimc_is_video_streamoff(file, vctx, type);
if (ret)
merr("fimc_is_video_streamoff is fail(%d)", vctx, ret);
return ret;
}
static int fimc_is_3aa_video_enum_input(struct file *file, void *priv,
struct v4l2_input *input)
{
/* Todo: add enum input control code */
return 0;
}
static int fimc_is_3aa_video_g_input(struct file *file, void *priv,
unsigned int *input)
{
/* Todo: add to get input control code */
return 0;
}
static int fimc_is_3aa_video_s_input(struct file *file, void *priv,
unsigned int input)
{
int ret = 0;
u32 stream, position, vindex, intype, leader;
struct fimc_is_video_ctx *vctx = file->private_data;
struct fimc_is_device_ischain *device;
FIMC_BUG(!vctx);
FIMC_BUG(!vctx->device);
device = GET_DEVICE(vctx);
stream = (input & INPUT_STREAM_MASK) >> INPUT_STREAM_SHIFT;
position = (input & INPUT_POSITION_MASK) >> INPUT_POSITION_SHIFT;
vindex = (input & INPUT_VINDEX_MASK) >> INPUT_VINDEX_SHIFT;
intype = (input & INPUT_INTYPE_MASK) >> INPUT_INTYPE_SHIFT;
leader = (input & INPUT_LEADER_MASK) >> INPUT_LEADER_SHIFT;
mdbgv_3aa("%s(input : %08X)[%d,%d,%d,%d,%d]\n", vctx, __func__, input,
stream, position, vindex, intype, leader);
ret = fimc_is_video_s_input(file, vctx);
if (ret) {
merr("fimc_is_video_s_input is fail(%d)", vctx, ret);
goto p_err;
}
ret = fimc_is_ischain_3aa_s_input(device, stream, position, vindex, intype, leader);
if (ret) {
merr("fimc_is_ischain_3aa_s_input is fail(%d)", vctx, ret);
goto p_err;
}
p_err:
return ret;
}
static int fimc_is_3aa_video_s_ctrl(struct file *file, void *priv,
struct v4l2_control *ctrl)
{
int ret = 0;
struct fimc_is_video_ctx *vctx = file->private_data;
struct fimc_is_device_ischain *device;
unsigned int value = 0;
unsigned int captureIntent = 0;
unsigned int captureCount = 0;
FIMC_BUG(!vctx);
FIMC_BUG(!GET_DEVICE(vctx));
FIMC_BUG(!ctrl);
mdbgv_3aa("%s\n", vctx, __func__);
device = GET_DEVICE(vctx);
switch (ctrl->id) {
case V4L2_CID_IS_INTENT:
value = (unsigned int)ctrl->value;
captureIntent = (value >> 16) & 0x0000FFFF;
switch (captureIntent) {
case AA_CAPTURE_INTENT_STILL_CAPTURE_DEBLUR_DYNAMIC_SHOT:
case AA_CAPTURE_INTENT_STILL_CAPTURE_OIS_DYNAMIC_SHOT:
case AA_CAPTURE_INTENT_STILL_CAPTURE_EXPOSURE_DYNAMIC_SHOT:
case AA_CAPTURE_INTENT_STILL_CAPTURE_MFHDR_DYNAMIC_SHOT:
case AA_CAPTURE_INTENT_STILL_CAPTURE_LLHDR_DYNAMIC_SHOT:
case AA_CAPTURE_INTENT_STILL_CAPTURE_SUPER_NIGHT_SHOT_HANDHELD:
case AA_CAPTURE_INTENT_STILL_CAPTURE_SUPER_NIGHT_SHOT_TRIPOD:
case AA_CAPTURE_INTENT_STILL_CAPTURE_CROPPED_REMOSAIC_DYNAMIC_SHOT:
captureCount = value & 0x0000FFFF;
break;
default:
captureIntent = ctrl->value;
captureCount = 0;
break;
}
device->group_3aa.intent_ctl.captureIntent = captureIntent;
device->group_3aa.intent_ctl.vendor_captureCount = captureCount;
if (captureIntent == AA_CAPTURE_INTENT_STILL_CAPTURE_OIS_MULTI) {
device->group_3aa.remainIntentCount = 2 + INTENT_RETRY_CNT;
} else {
device->group_3aa.remainIntentCount = 0 + INTENT_RETRY_CNT;
}
minfo("[3AA:V] s_ctrl intent(%d) count(%d) remainIntentCount(%d)\n",
vctx, captureIntent, captureCount, device->group_3aa.remainIntentCount);
break;
case V4L2_CID_IS_FORCE_DONE:
set_bit(FIMC_IS_GROUP_REQUEST_FSTOP, &device->group_3aa.state);
break;
default:
ret = fimc_is_video_s_ctrl(file, vctx, ctrl);
if (ret) {
merr("fimc_is_video_s_ctrl is fail(%d)", device, ret);
goto p_err;
}
break;
}
p_err:
return ret;
}
static int fimc_is_3aa_video_g_ctrl(struct file *file, void *priv,
struct v4l2_control *ctrl)
{
/* Todo: add to get control code */
return 0;
}
static int fimc_is_3aa_video_s_ext_ctrl(struct file *file, void *priv,
struct v4l2_ext_controls *ctrls)
{
int ret = 0;
int i;
struct fimc_is_video_ctx *vctx = file->private_data;
struct fimc_is_device_ischain *device;
struct fimc_is_framemgr *framemgr;
struct fimc_is_queue *queue;
struct v4l2_ext_control *ext_ctrl;
struct v4l2_control ctrl;
struct nfd_info *nfd_data, local_nfd_info;
unsigned long flags;
FIMC_BUG(!vctx);
FIMC_BUG(!GET_DEVICE(vctx));
FIMC_BUG(!ctrls);
mdbgv_3aa("%s\n", vctx, __func__);
if (ctrls->which != V4L2_CTRL_CLASS_CAMERA) {
merr("Invalid control class(%d)", vctx, ctrls->which);
ret = -EINVAL;
goto p_err;
}
device = GET_DEVICE(vctx);
queue = GET_QUEUE(vctx);
framemgr = &queue->framemgr;
for (i = 0; i < ctrls->count; i++) {
ext_ctrl = (ctrls->controls + i);
switch (ext_ctrl->id) {
#ifdef ENABLE_ULTRA_FAST_SHOT
case V4L2_CID_IS_FAST_CAPTURE_CONTROL:
{
struct fast_ctl_capture *fast_capture =
(struct fast_ctl_capture *)&device->is_region->fast_ctl.fast_capture;
ret = copy_from_user(fast_capture, ext_ctrl->ptr, sizeof(struct fast_ctl_capture));
if (ret) {
merr("copy_from_user is fail(%d)", vctx, ret);
goto p_err;
}
fast_capture->ready = 1;
device->fastctlmgr.fast_capture_count = 2;
vb2_ion_sync_for_device(
device->imemory.fw_cookie,
(ulong)fast_capture - device->imemory.kvaddr,
sizeof(struct fast_ctl_capture),
DMA_TO_DEVICE);
mvinfo("Fast capture control(Intent:%d, count:%d, exposureTime:%d)\n",
vctx, vctx->video, fast_capture->capture_intent, fast_capture->capture_count,
fast_capture->capture_exposureTime);
}
break;
#endif
case V4L2_CID_SENSOR_SET_CAPTURE_INTENT_INFO:
{
struct fimc_is_group *head;
struct capture_intent_info_t info;
ret = copy_from_user(&info, ext_ctrl->ptr, sizeof(struct capture_intent_info_t));
if (ret) {
err("fail to copy_from_user, ret(%d)\n", ret);
goto p_err;
}
head = GET_HEAD_GROUP_IN_DEVICE(FIMC_IS_DEVICE_ISCHAIN, (&device->group_3aa));
head->intent_ctl.captureIntent = info.captureIntent;
head->intent_ctl.vendor_captureCount = info.captureCount;
if (info.captureIntent == AA_CAPTURE_INTENT_STILL_CAPTURE_OIS_MULTI) {
head->remainIntentCount = 2 + INTENT_RETRY_CNT;
} else {
head->remainIntentCount = 0 + INTENT_RETRY_CNT;
}
info("s_ext_ctrl SET_CAPTURE_INTENT_INFO, intent(%d) count(%d) remainIntentCount(%d)\n",
info.captureIntent, info.captureCount, head->remainIntentCount);
break;
}
case V4L2_CID_IS_S_NFD_DATA:
nfd_data = (struct nfd_info *)&device->is_region->fd_info;
ret = copy_from_user(&local_nfd_info, ext_ctrl->ptr, sizeof(struct nfd_info));
spin_lock_irqsave(&device->is_region->fd_info_slock, flags);
memcpy((void *)nfd_data, &local_nfd_info, sizeof(struct nfd_info));
if ((nfd_data->face_num < 0) || (nfd_data->face_num > MAX_FACE_COUNT))
nfd_data->face_num = 0;
spin_unlock_irqrestore(&device->is_region->fd_info_slock, flags);
if (ret) {
merr("copy_from_user of nfd_info is fail(%d)", vctx, ret);
goto p_err;
}
mdbgv_3aa("[F%d] face num(%d)\n", vctx,
nfd_data->frame_count, nfd_data->face_num);
break;
default:
ctrl.id = ext_ctrl->id;
ctrl.value = ext_ctrl->value;
ret = fimc_is_video_s_ctrl(file, vctx, &ctrl);
if (ret) {
merr("fimc_is_video_s_ctrl is fail(%d)", device, ret);
goto p_err;
}
break;
}
}
p_err:
return ret;
}
static int fimc_is_3aa_video_g_ext_ctrl(struct file *file, void *priv,
struct v4l2_ext_controls *ctrls)
{
/* Todo: add to get extra control code */
return 0;
}
const struct v4l2_ioctl_ops fimc_is_3aa_video_ioctl_ops = {
.vidioc_querycap = fimc_is_3aa_video_querycap,
.vidioc_enum_fmt_vid_out_mplane = fimc_is_3aa_video_enum_fmt_mplane,
.vidioc_enum_fmt_vid_cap_mplane = fimc_is_3aa_video_enum_fmt_mplane,
.vidioc_g_fmt_vid_out_mplane = fimc_is_3aa_video_get_format_mplane,
.vidioc_g_fmt_vid_cap_mplane = fimc_is_3aa_video_get_format_mplane,
.vidioc_s_fmt_vid_out_mplane = fimc_is_3aa_video_set_format_mplane,
.vidioc_s_fmt_vid_cap_mplane = fimc_is_3aa_video_set_format_mplane,
.vidioc_querybuf = fimc_is_3aa_video_querybuf,
.vidioc_reqbufs = fimc_is_3aa_video_reqbufs,
.vidioc_qbuf = fimc_is_3aa_video_qbuf,
.vidioc_dqbuf = fimc_is_3aa_video_dqbuf,
.vidioc_prepare_buf = fimc_is_3aa_video_prepare,
.vidioc_streamon = fimc_is_3aa_video_streamon,
.vidioc_streamoff = fimc_is_3aa_video_streamoff,
.vidioc_enum_input = fimc_is_3aa_video_enum_input,
.vidioc_g_input = fimc_is_3aa_video_g_input,
.vidioc_s_input = fimc_is_3aa_video_s_input,
.vidioc_s_ctrl = fimc_is_3aa_video_s_ctrl,
.vidioc_g_ctrl = fimc_is_3aa_video_g_ctrl,
.vidioc_s_ext_ctrls = fimc_is_3aa_video_s_ext_ctrl,
.vidioc_g_ext_ctrls = fimc_is_3aa_video_g_ext_ctrl,
.vidioc_cropcap = fimc_is_3aa_video_cropcap,
.vidioc_g_crop = fimc_is_3aa_video_get_crop,
.vidioc_s_crop = fimc_is_3aa_video_set_crop,
};
static int fimc_is_3aa_queue_setup(struct vb2_queue *vbq,
unsigned int *num_buffers,
unsigned int *num_planes,
unsigned int sizes[],
struct device *alloc_devs[])
{
int ret = 0;
struct fimc_is_video_ctx *vctx = vbq->drv_priv;
struct fimc_is_video *video;
struct fimc_is_queue *queue;
FIMC_BUG(!vctx);
FIMC_BUG(!vctx->video);
mdbgv_3aa("%s\n", vctx, __func__);
video = GET_VIDEO(vctx);
queue = GET_QUEUE(vctx);
ret = fimc_is_queue_setup(queue,
video->alloc_ctx,
num_planes,
sizes,
alloc_devs);
if (ret)
merr("fimc_is_queue_setup is fail(%d)", vctx, ret);
return ret;
}
static int fimc_is_3aa_start_streaming(struct vb2_queue *vbq,
unsigned int count)
{
int ret = 0;
struct fimc_is_video_ctx *vctx = vbq->drv_priv;
struct fimc_is_queue *queue;
struct fimc_is_device_ischain *device;
FIMC_BUG(!vctx);
FIMC_BUG(!GET_DEVICE(vctx));
mdbgv_3aa("%s\n", vctx, __func__);
device = GET_DEVICE(vctx);
queue = GET_QUEUE(vctx);
ret = fimc_is_queue_start_streaming(queue, device);
if (ret) {
merr("fimc_is_queue_start_streaming is fail(%d)", device, ret);
goto p_err;
}
p_err:
return ret;
}
static void fimc_is_3aa_stop_streaming(struct vb2_queue *vbq)
{
int ret = 0;
struct fimc_is_video_ctx *vctx = vbq->drv_priv;
struct fimc_is_queue *queue;
struct fimc_is_device_ischain *device;
FIMC_BUG_VOID(!vctx);
FIMC_BUG_VOID(!GET_DEVICE(vctx));
mdbgv_3aa("%s\n", vctx, __func__);
device = GET_DEVICE(vctx);
queue = GET_QUEUE(vctx);
ret = fimc_is_queue_stop_streaming(queue, device);
if (ret) {
merr("fimc_is_queue_stop_streaming is fail(%d)", device, ret);
return;
}
}
static void fimc_is_3aa_buffer_queue(struct vb2_buffer *vb)
{
int ret = 0;
struct fimc_is_video_ctx *vctx = vb->vb2_queue->drv_priv;
struct fimc_is_device_ischain *device;
struct fimc_is_queue *queue;
FIMC_BUG_VOID(!vctx);
FIMC_BUG_VOID(!GET_DEVICE(vctx));
mvdbgs(3, "%s(%d)\n", vctx, &vctx->queue, __func__, vb->index);
device = GET_DEVICE(vctx);
queue = GET_QUEUE(vctx);
ret = fimc_is_queue_buffer_queue(queue, vb);
if (ret) {
merr("fimc_is_queue_buffer_queue is fail(%d)", device, ret);
return;
}
ret = fimc_is_ischain_3aa_buffer_queue(device, queue, vb->index);
if (ret) {
merr("fimc_is_ischain_3aa_buffer_queue is fail(%d)", device, ret);
return;
}
}
static void fimc_is_3aa_buffer_finish(struct vb2_buffer *vb)
{
int ret = 0;
struct fimc_is_video_ctx *vctx = vb->vb2_queue->drv_priv;
struct fimc_is_device_ischain *device;
FIMC_BUG_VOID(!vctx);
FIMC_BUG_VOID(!GET_DEVICE(vctx));
mvdbgs(3, "%s(%d)\n", vctx, &vctx->queue, __func__, vb->index);
device = GET_DEVICE(vctx);
fimc_is_queue_buffer_finish(vb);
ret = fimc_is_ischain_3aa_buffer_finish(device, vb->index);
if (ret) {
merr("fimc_is_ischain_3aa_buffer_finish is fail(%d)", device, ret);
return;
}
}
const struct vb2_ops fimc_is_3aa_qops = {
.queue_setup = fimc_is_3aa_queue_setup,
.buf_init = fimc_is_queue_buffer_init,
.buf_cleanup = fimc_is_queue_buffer_cleanup,
.buf_prepare = fimc_is_queue_buffer_prepare,
.buf_queue = fimc_is_3aa_buffer_queue,
.buf_finish = fimc_is_3aa_buffer_finish,
.wait_prepare = fimc_is_queue_wait_prepare,
.wait_finish = fimc_is_queue_wait_finish,
.start_streaming = fimc_is_3aa_start_streaming,
.stop_streaming = fimc_is_3aa_stop_streaming,
};