The ISP8000Nano v18.02 (found in the i.MX8MP) requires the ISP to be enabled (as indicated by the ISP_CTRL.ISP_ENABLE bit) to configure the lens shading table in internal RAM. The driver currently configures all ISP initial parameters before enabling the ISP, which causes the LSC RAM to not be initialized properly. To fix this, split the rkisp1_params_configure() function into a rkisp1_params_pre_configure() and a rkisp1_params_post_configure(). The former configures all ISP parameters but LSC, while the latter configures LSC. To implement this, the rkisp1_params_apply_params_cfg() function is deconstructed, with two small helpers created to deal with the parameters buffers, which are then used in rkisp1_params_isr(), rkisp1_params_pre_configure() and rkisp1_params_post_configure(). While this initialization ordering is only needed for the ISP8000Nano v18.02, it doesn't affect other ISP versions negatively, and can thus be followed unconditionally. Signed-off-by: Laurent Pinchart <laurent.pinchart@ideasonboard.com> Reviewed-by: Dafna Hirschfeld <dafna@fastmail.com> Signed-off-by: Mauro Carvalho Chehab <mchehab@kernel.org>
1059 lines
31 KiB
C
1059 lines
31 KiB
C
// SPDX-License-Identifier: (GPL-2.0+ OR MIT)
|
|
/*
|
|
* Rockchip ISP1 Driver - ISP Subdevice
|
|
*
|
|
* Copyright (C) 2019 Collabora, Ltd.
|
|
*
|
|
* Based on Rockchip ISP1 driver by Rockchip Electronics Co., Ltd.
|
|
* Copyright (C) 2017 Rockchip Electronics Co., Ltd.
|
|
*/
|
|
|
|
#include <linux/iopoll.h>
|
|
#include <linux/pm_runtime.h>
|
|
#include <linux/videodev2.h>
|
|
#include <linux/vmalloc.h>
|
|
|
|
#include <media/v4l2-event.h>
|
|
|
|
#include "rkisp1-common.h"
|
|
|
|
#define RKISP1_DEF_SINK_PAD_FMT MEDIA_BUS_FMT_SRGGB10_1X10
|
|
#define RKISP1_DEF_SRC_PAD_FMT MEDIA_BUS_FMT_YUYV8_2X8
|
|
|
|
#define RKISP1_ISP_DEV_NAME RKISP1_DRIVER_NAME "_isp"
|
|
|
|
/*
|
|
* NOTE: MIPI controller and input MUX are also configured in this file.
|
|
* This is because ISP Subdev describes not only ISP submodule (input size,
|
|
* format, output size, format), but also a virtual route device.
|
|
*/
|
|
|
|
/*
|
|
* There are many variables named with format/frame in below code,
|
|
* please see here for their meaning.
|
|
* Cropping in the sink pad defines the image region from the sensor.
|
|
* Cropping in the source pad defines the region for the Image Stabilizer (IS)
|
|
*
|
|
* Cropping regions of ISP
|
|
*
|
|
* +---------------------------------------------------------+
|
|
* | Sensor image |
|
|
* | +---------------------------------------------------+ |
|
|
* | | CIF_ISP_ACQ (for black level) | |
|
|
* | | sink pad format | |
|
|
* | | +--------------------------------------------+ | |
|
|
* | | | CIF_ISP_OUT | | |
|
|
* | | | sink pad crop | | |
|
|
* | | | +---------------------------------+ | | |
|
|
* | | | | CIF_ISP_IS | | | |
|
|
* | | | | source pad crop and format | | | |
|
|
* | | | +---------------------------------+ | | |
|
|
* | | +--------------------------------------------+ | |
|
|
* | +---------------------------------------------------+ |
|
|
* +---------------------------------------------------------+
|
|
*/
|
|
|
|
/* ----------------------------------------------------------------------------
|
|
* Helpers
|
|
*/
|
|
|
|
static struct v4l2_mbus_framefmt *
|
|
rkisp1_isp_get_pad_fmt(struct rkisp1_isp *isp,
|
|
struct v4l2_subdev_state *sd_state,
|
|
unsigned int pad, u32 which)
|
|
{
|
|
struct v4l2_subdev_state state = {
|
|
.pads = isp->pad_cfg
|
|
};
|
|
|
|
if (which == V4L2_SUBDEV_FORMAT_TRY)
|
|
return v4l2_subdev_get_try_format(&isp->sd, sd_state, pad);
|
|
else
|
|
return v4l2_subdev_get_try_format(&isp->sd, &state, pad);
|
|
}
|
|
|
|
static struct v4l2_rect *
|
|
rkisp1_isp_get_pad_crop(struct rkisp1_isp *isp,
|
|
struct v4l2_subdev_state *sd_state,
|
|
unsigned int pad, u32 which)
|
|
{
|
|
struct v4l2_subdev_state state = {
|
|
.pads = isp->pad_cfg
|
|
};
|
|
|
|
if (which == V4L2_SUBDEV_FORMAT_TRY)
|
|
return v4l2_subdev_get_try_crop(&isp->sd, sd_state, pad);
|
|
else
|
|
return v4l2_subdev_get_try_crop(&isp->sd, &state, pad);
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------------
|
|
* Camera Interface registers configurations
|
|
*/
|
|
|
|
/*
|
|
* Image Stabilization.
|
|
* This should only be called when configuring CIF
|
|
* or at the frame end interrupt
|
|
*/
|
|
static void rkisp1_config_ism(struct rkisp1_isp *isp)
|
|
{
|
|
const struct v4l2_rect *src_crop =
|
|
rkisp1_isp_get_pad_crop(isp, NULL,
|
|
RKISP1_ISP_PAD_SOURCE_VIDEO,
|
|
V4L2_SUBDEV_FORMAT_ACTIVE);
|
|
struct rkisp1_device *rkisp1 = isp->rkisp1;
|
|
u32 val;
|
|
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_IS_RECENTER, 0);
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_IS_MAX_DX, 0);
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_IS_MAX_DY, 0);
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_IS_DISPLACE, 0);
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_IS_H_OFFS, src_crop->left);
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_IS_V_OFFS, src_crop->top);
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_IS_H_SIZE, src_crop->width);
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_IS_V_SIZE, src_crop->height);
|
|
|
|
/* IS(Image Stabilization) is always on, working as output crop */
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_IS_CTRL, 1);
|
|
val = rkisp1_read(rkisp1, RKISP1_CIF_ISP_CTRL);
|
|
val |= RKISP1_CIF_ISP_CTRL_ISP_CFG_UPD;
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_CTRL, val);
|
|
}
|
|
|
|
/*
|
|
* configure ISP blocks with input format, size......
|
|
*/
|
|
static int rkisp1_config_isp(struct rkisp1_isp *isp,
|
|
enum v4l2_mbus_type mbus_type, u32 mbus_flags)
|
|
{
|
|
struct rkisp1_device *rkisp1 = isp->rkisp1;
|
|
u32 isp_ctrl = 0, irq_mask = 0, acq_mult = 0, acq_prop = 0;
|
|
const struct rkisp1_mbus_info *sink_fmt = isp->sink_fmt;
|
|
const struct rkisp1_mbus_info *src_fmt = isp->src_fmt;
|
|
const struct v4l2_mbus_framefmt *sink_frm;
|
|
const struct v4l2_rect *sink_crop;
|
|
|
|
sink_frm = rkisp1_isp_get_pad_fmt(isp, NULL,
|
|
RKISP1_ISP_PAD_SINK_VIDEO,
|
|
V4L2_SUBDEV_FORMAT_ACTIVE);
|
|
sink_crop = rkisp1_isp_get_pad_crop(isp, NULL,
|
|
RKISP1_ISP_PAD_SINK_VIDEO,
|
|
V4L2_SUBDEV_FORMAT_ACTIVE);
|
|
|
|
if (sink_fmt->pixel_enc == V4L2_PIXEL_ENC_BAYER) {
|
|
acq_mult = 1;
|
|
if (src_fmt->pixel_enc == V4L2_PIXEL_ENC_BAYER) {
|
|
if (mbus_type == V4L2_MBUS_BT656)
|
|
isp_ctrl = RKISP1_CIF_ISP_CTRL_ISP_MODE_RAW_PICT_ITU656;
|
|
else
|
|
isp_ctrl = RKISP1_CIF_ISP_CTRL_ISP_MODE_RAW_PICT;
|
|
} else {
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_DEMOSAIC,
|
|
RKISP1_CIF_ISP_DEMOSAIC_TH(0xc));
|
|
|
|
if (mbus_type == V4L2_MBUS_BT656)
|
|
isp_ctrl = RKISP1_CIF_ISP_CTRL_ISP_MODE_BAYER_ITU656;
|
|
else
|
|
isp_ctrl = RKISP1_CIF_ISP_CTRL_ISP_MODE_BAYER_ITU601;
|
|
}
|
|
} else if (sink_fmt->pixel_enc == V4L2_PIXEL_ENC_YUV) {
|
|
acq_mult = 2;
|
|
if (mbus_type == V4L2_MBUS_CSI2_DPHY) {
|
|
isp_ctrl = RKISP1_CIF_ISP_CTRL_ISP_MODE_ITU601;
|
|
} else {
|
|
if (mbus_type == V4L2_MBUS_BT656)
|
|
isp_ctrl = RKISP1_CIF_ISP_CTRL_ISP_MODE_ITU656;
|
|
else
|
|
isp_ctrl = RKISP1_CIF_ISP_CTRL_ISP_MODE_ITU601;
|
|
}
|
|
|
|
irq_mask |= RKISP1_CIF_ISP_DATA_LOSS;
|
|
}
|
|
|
|
/* Set up input acquisition properties */
|
|
if (mbus_type == V4L2_MBUS_BT656 || mbus_type == V4L2_MBUS_PARALLEL) {
|
|
if (mbus_flags & V4L2_MBUS_PCLK_SAMPLE_RISING)
|
|
acq_prop |= RKISP1_CIF_ISP_ACQ_PROP_POS_EDGE;
|
|
|
|
switch (sink_fmt->bus_width) {
|
|
case 8:
|
|
acq_prop |= RKISP1_CIF_ISP_ACQ_PROP_IN_SEL_8B_ZERO;
|
|
break;
|
|
case 10:
|
|
acq_prop |= RKISP1_CIF_ISP_ACQ_PROP_IN_SEL_10B_ZERO;
|
|
break;
|
|
case 12:
|
|
acq_prop |= RKISP1_CIF_ISP_ACQ_PROP_IN_SEL_12B;
|
|
break;
|
|
default:
|
|
dev_err(rkisp1->dev, "Invalid bus width %u\n",
|
|
sink_fmt->bus_width);
|
|
return -EINVAL;
|
|
}
|
|
}
|
|
|
|
if (mbus_type == V4L2_MBUS_PARALLEL) {
|
|
if (mbus_flags & V4L2_MBUS_VSYNC_ACTIVE_LOW)
|
|
acq_prop |= RKISP1_CIF_ISP_ACQ_PROP_VSYNC_LOW;
|
|
|
|
if (mbus_flags & V4L2_MBUS_HSYNC_ACTIVE_LOW)
|
|
acq_prop |= RKISP1_CIF_ISP_ACQ_PROP_HSYNC_LOW;
|
|
}
|
|
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_CTRL, isp_ctrl);
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_ACQ_PROP,
|
|
acq_prop | sink_fmt->yuv_seq |
|
|
RKISP1_CIF_ISP_ACQ_PROP_BAYER_PAT(sink_fmt->bayer_pat) |
|
|
RKISP1_CIF_ISP_ACQ_PROP_FIELD_SEL_ALL);
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_ACQ_NR_FRAMES, 0);
|
|
|
|
/* Acquisition Size */
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_ACQ_H_OFFS, 0);
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_ACQ_V_OFFS, 0);
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_ACQ_H_SIZE,
|
|
acq_mult * sink_frm->width);
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_ACQ_V_SIZE, sink_frm->height);
|
|
|
|
/* ISP Out Area */
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_OUT_H_OFFS, sink_crop->left);
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_OUT_V_OFFS, sink_crop->top);
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_OUT_H_SIZE, sink_crop->width);
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_OUT_V_SIZE, sink_crop->height);
|
|
|
|
irq_mask |= RKISP1_CIF_ISP_FRAME | RKISP1_CIF_ISP_V_START |
|
|
RKISP1_CIF_ISP_PIC_SIZE_ERROR;
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_IMSC, irq_mask);
|
|
|
|
if (src_fmt->pixel_enc == V4L2_PIXEL_ENC_BAYER) {
|
|
rkisp1_params_disable(&rkisp1->params);
|
|
} else {
|
|
struct v4l2_mbus_framefmt *src_frm;
|
|
|
|
src_frm = rkisp1_isp_get_pad_fmt(isp, NULL,
|
|
RKISP1_ISP_PAD_SOURCE_VIDEO,
|
|
V4L2_SUBDEV_FORMAT_ACTIVE);
|
|
rkisp1_params_pre_configure(&rkisp1->params, sink_fmt->bayer_pat,
|
|
src_frm->quantization,
|
|
src_frm->ycbcr_enc);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* Configure MUX */
|
|
static void rkisp1_config_path(struct rkisp1_isp *isp,
|
|
enum v4l2_mbus_type mbus_type)
|
|
{
|
|
struct rkisp1_device *rkisp1 = isp->rkisp1;
|
|
u32 dpcl = rkisp1_read(rkisp1, RKISP1_CIF_VI_DPCL);
|
|
|
|
if (mbus_type == V4L2_MBUS_BT656 || mbus_type == V4L2_MBUS_PARALLEL)
|
|
dpcl |= RKISP1_CIF_VI_DPCL_IF_SEL_PARALLEL;
|
|
else if (mbus_type == V4L2_MBUS_CSI2_DPHY)
|
|
dpcl |= RKISP1_CIF_VI_DPCL_IF_SEL_MIPI;
|
|
|
|
rkisp1_write(rkisp1, RKISP1_CIF_VI_DPCL, dpcl);
|
|
}
|
|
|
|
/* Hardware configure Entry */
|
|
static int rkisp1_config_cif(struct rkisp1_isp *isp,
|
|
enum v4l2_mbus_type mbus_type, u32 mbus_flags)
|
|
{
|
|
int ret;
|
|
|
|
ret = rkisp1_config_isp(isp, mbus_type, mbus_flags);
|
|
if (ret)
|
|
return ret;
|
|
|
|
rkisp1_config_path(isp, mbus_type);
|
|
rkisp1_config_ism(isp);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void rkisp1_isp_stop(struct rkisp1_isp *isp)
|
|
{
|
|
struct rkisp1_device *rkisp1 = isp->rkisp1;
|
|
u32 val;
|
|
|
|
/*
|
|
* ISP(mi) stop in mi frame end -> Stop ISP(mipi) ->
|
|
* Stop ISP(isp) ->wait for ISP isp off
|
|
*/
|
|
/* stop and clear MI and ISP interrupts */
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_IMSC, 0);
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_ICR, ~0);
|
|
|
|
rkisp1_write(rkisp1, RKISP1_CIF_MI_IMSC, 0);
|
|
rkisp1_write(rkisp1, RKISP1_CIF_MI_ICR, ~0);
|
|
|
|
/* stop ISP */
|
|
val = rkisp1_read(rkisp1, RKISP1_CIF_ISP_CTRL);
|
|
val &= ~(RKISP1_CIF_ISP_CTRL_ISP_INFORM_ENABLE |
|
|
RKISP1_CIF_ISP_CTRL_ISP_ENABLE);
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_CTRL, val);
|
|
|
|
val = rkisp1_read(rkisp1, RKISP1_CIF_ISP_CTRL);
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_CTRL,
|
|
val | RKISP1_CIF_ISP_CTRL_ISP_CFG_UPD);
|
|
|
|
readx_poll_timeout(readl, rkisp1->base_addr + RKISP1_CIF_ISP_RIS,
|
|
val, val & RKISP1_CIF_ISP_OFF, 20, 100);
|
|
rkisp1_write(rkisp1, RKISP1_CIF_VI_IRCL,
|
|
RKISP1_CIF_VI_IRCL_MIPI_SW_RST |
|
|
RKISP1_CIF_VI_IRCL_ISP_SW_RST);
|
|
rkisp1_write(rkisp1, RKISP1_CIF_VI_IRCL, 0x0);
|
|
}
|
|
|
|
static void rkisp1_config_clk(struct rkisp1_isp *isp)
|
|
{
|
|
struct rkisp1_device *rkisp1 = isp->rkisp1;
|
|
|
|
u32 val = RKISP1_CIF_VI_ICCL_ISP_CLK | RKISP1_CIF_VI_ICCL_CP_CLK |
|
|
RKISP1_CIF_VI_ICCL_MRSZ_CLK | RKISP1_CIF_VI_ICCL_SRSZ_CLK |
|
|
RKISP1_CIF_VI_ICCL_JPEG_CLK | RKISP1_CIF_VI_ICCL_MI_CLK |
|
|
RKISP1_CIF_VI_ICCL_IE_CLK | RKISP1_CIF_VI_ICCL_MIPI_CLK |
|
|
RKISP1_CIF_VI_ICCL_DCROP_CLK;
|
|
|
|
rkisp1_write(rkisp1, RKISP1_CIF_VI_ICCL, val);
|
|
|
|
/* ensure sp and mp can run at the same time in V12 */
|
|
if (rkisp1->info->isp_ver == RKISP1_V12) {
|
|
val = RKISP1_CIF_CLK_CTRL_MI_Y12 | RKISP1_CIF_CLK_CTRL_MI_SP |
|
|
RKISP1_CIF_CLK_CTRL_MI_RAW0 | RKISP1_CIF_CLK_CTRL_MI_RAW1 |
|
|
RKISP1_CIF_CLK_CTRL_MI_READ | RKISP1_CIF_CLK_CTRL_MI_RAWRD |
|
|
RKISP1_CIF_CLK_CTRL_CP | RKISP1_CIF_CLK_CTRL_IE;
|
|
rkisp1_write(rkisp1, RKISP1_CIF_VI_ISP_CLK_CTRL_V12, val);
|
|
}
|
|
}
|
|
|
|
static void rkisp1_isp_start(struct rkisp1_isp *isp)
|
|
{
|
|
struct rkisp1_device *rkisp1 = isp->rkisp1;
|
|
u32 val;
|
|
|
|
rkisp1_config_clk(isp);
|
|
|
|
/* Activate ISP */
|
|
val = rkisp1_read(rkisp1, RKISP1_CIF_ISP_CTRL);
|
|
val |= RKISP1_CIF_ISP_CTRL_ISP_CFG_UPD |
|
|
RKISP1_CIF_ISP_CTRL_ISP_ENABLE |
|
|
RKISP1_CIF_ISP_CTRL_ISP_INFORM_ENABLE;
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_CTRL, val);
|
|
|
|
if (isp->src_fmt->pixel_enc != V4L2_PIXEL_ENC_BAYER)
|
|
rkisp1_params_post_configure(&rkisp1->params);
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------------
|
|
* Subdev pad operations
|
|
*/
|
|
|
|
static inline struct rkisp1_isp *to_rkisp1_isp(struct v4l2_subdev *sd)
|
|
{
|
|
return container_of(sd, struct rkisp1_isp, sd);
|
|
}
|
|
|
|
static int rkisp1_isp_enum_mbus_code(struct v4l2_subdev *sd,
|
|
struct v4l2_subdev_state *sd_state,
|
|
struct v4l2_subdev_mbus_code_enum *code)
|
|
{
|
|
unsigned int i, dir;
|
|
int pos = 0;
|
|
|
|
if (code->pad == RKISP1_ISP_PAD_SINK_VIDEO) {
|
|
dir = RKISP1_ISP_SD_SINK;
|
|
} else if (code->pad == RKISP1_ISP_PAD_SOURCE_VIDEO) {
|
|
dir = RKISP1_ISP_SD_SRC;
|
|
} else {
|
|
if (code->index > 0)
|
|
return -EINVAL;
|
|
code->code = MEDIA_BUS_FMT_METADATA_FIXED;
|
|
return 0;
|
|
}
|
|
|
|
for (i = 0; ; i++) {
|
|
const struct rkisp1_mbus_info *fmt =
|
|
rkisp1_mbus_info_get_by_index(i);
|
|
|
|
if (!fmt)
|
|
return -EINVAL;
|
|
|
|
if (fmt->direction & dir)
|
|
pos++;
|
|
|
|
if (code->index == pos - 1) {
|
|
code->code = fmt->mbus_code;
|
|
if (fmt->pixel_enc == V4L2_PIXEL_ENC_YUV &&
|
|
dir == RKISP1_ISP_SD_SRC)
|
|
code->flags =
|
|
V4L2_SUBDEV_MBUS_CODE_CSC_QUANTIZATION;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
return -EINVAL;
|
|
}
|
|
|
|
static int rkisp1_isp_enum_frame_size(struct v4l2_subdev *sd,
|
|
struct v4l2_subdev_state *sd_state,
|
|
struct v4l2_subdev_frame_size_enum *fse)
|
|
{
|
|
const struct rkisp1_mbus_info *mbus_info;
|
|
|
|
if (fse->pad == RKISP1_ISP_PAD_SINK_PARAMS ||
|
|
fse->pad == RKISP1_ISP_PAD_SOURCE_STATS)
|
|
return -ENOTTY;
|
|
|
|
if (fse->index > 0)
|
|
return -EINVAL;
|
|
|
|
mbus_info = rkisp1_mbus_info_get_by_code(fse->code);
|
|
if (!mbus_info)
|
|
return -EINVAL;
|
|
|
|
if (!(mbus_info->direction & RKISP1_ISP_SD_SINK) &&
|
|
fse->pad == RKISP1_ISP_PAD_SINK_VIDEO)
|
|
return -EINVAL;
|
|
|
|
if (!(mbus_info->direction & RKISP1_ISP_SD_SRC) &&
|
|
fse->pad == RKISP1_ISP_PAD_SOURCE_VIDEO)
|
|
return -EINVAL;
|
|
|
|
fse->min_width = RKISP1_ISP_MIN_WIDTH;
|
|
fse->max_width = RKISP1_ISP_MAX_WIDTH;
|
|
fse->min_height = RKISP1_ISP_MIN_HEIGHT;
|
|
fse->max_height = RKISP1_ISP_MAX_HEIGHT;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int rkisp1_isp_init_config(struct v4l2_subdev *sd,
|
|
struct v4l2_subdev_state *sd_state)
|
|
{
|
|
struct v4l2_mbus_framefmt *sink_fmt, *src_fmt;
|
|
struct v4l2_rect *sink_crop, *src_crop;
|
|
|
|
/* Video. */
|
|
sink_fmt = v4l2_subdev_get_try_format(sd, sd_state,
|
|
RKISP1_ISP_PAD_SINK_VIDEO);
|
|
sink_fmt->width = RKISP1_DEFAULT_WIDTH;
|
|
sink_fmt->height = RKISP1_DEFAULT_HEIGHT;
|
|
sink_fmt->field = V4L2_FIELD_NONE;
|
|
sink_fmt->code = RKISP1_DEF_SINK_PAD_FMT;
|
|
sink_fmt->colorspace = V4L2_COLORSPACE_RAW;
|
|
sink_fmt->xfer_func = V4L2_XFER_FUNC_NONE;
|
|
sink_fmt->ycbcr_enc = V4L2_YCBCR_ENC_601;
|
|
sink_fmt->quantization = V4L2_QUANTIZATION_FULL_RANGE;
|
|
|
|
sink_crop = v4l2_subdev_get_try_crop(sd, sd_state,
|
|
RKISP1_ISP_PAD_SINK_VIDEO);
|
|
sink_crop->width = RKISP1_DEFAULT_WIDTH;
|
|
sink_crop->height = RKISP1_DEFAULT_HEIGHT;
|
|
sink_crop->left = 0;
|
|
sink_crop->top = 0;
|
|
|
|
src_fmt = v4l2_subdev_get_try_format(sd, sd_state,
|
|
RKISP1_ISP_PAD_SOURCE_VIDEO);
|
|
*src_fmt = *sink_fmt;
|
|
src_fmt->code = RKISP1_DEF_SRC_PAD_FMT;
|
|
src_fmt->colorspace = V4L2_COLORSPACE_SRGB;
|
|
src_fmt->xfer_func = V4L2_XFER_FUNC_SRGB;
|
|
src_fmt->ycbcr_enc = V4L2_YCBCR_ENC_601;
|
|
src_fmt->quantization = V4L2_QUANTIZATION_LIM_RANGE;
|
|
|
|
src_crop = v4l2_subdev_get_try_crop(sd, sd_state,
|
|
RKISP1_ISP_PAD_SOURCE_VIDEO);
|
|
*src_crop = *sink_crop;
|
|
|
|
/* Parameters and statistics. */
|
|
sink_fmt = v4l2_subdev_get_try_format(sd, sd_state,
|
|
RKISP1_ISP_PAD_SINK_PARAMS);
|
|
src_fmt = v4l2_subdev_get_try_format(sd, sd_state,
|
|
RKISP1_ISP_PAD_SOURCE_STATS);
|
|
sink_fmt->width = 0;
|
|
sink_fmt->height = 0;
|
|
sink_fmt->field = V4L2_FIELD_NONE;
|
|
sink_fmt->code = MEDIA_BUS_FMT_METADATA_FIXED;
|
|
*src_fmt = *sink_fmt;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void rkisp1_isp_set_src_fmt(struct rkisp1_isp *isp,
|
|
struct v4l2_subdev_state *sd_state,
|
|
struct v4l2_mbus_framefmt *format,
|
|
unsigned int which)
|
|
{
|
|
const struct rkisp1_mbus_info *sink_info;
|
|
const struct rkisp1_mbus_info *src_info;
|
|
struct v4l2_mbus_framefmt *sink_fmt;
|
|
struct v4l2_mbus_framefmt *src_fmt;
|
|
const struct v4l2_rect *src_crop;
|
|
bool set_csc;
|
|
|
|
sink_fmt = rkisp1_isp_get_pad_fmt(isp, sd_state,
|
|
RKISP1_ISP_PAD_SINK_VIDEO, which);
|
|
src_fmt = rkisp1_isp_get_pad_fmt(isp, sd_state,
|
|
RKISP1_ISP_PAD_SOURCE_VIDEO, which);
|
|
src_crop = rkisp1_isp_get_pad_crop(isp, sd_state,
|
|
RKISP1_ISP_PAD_SOURCE_VIDEO, which);
|
|
|
|
/*
|
|
* Media bus code. The ISP can operate in pass-through mode (Bayer in,
|
|
* Bayer out or YUV in, YUV out) or process Bayer data to YUV, but
|
|
* can't convert from YUV to Bayer.
|
|
*/
|
|
sink_info = rkisp1_mbus_info_get_by_code(sink_fmt->code);
|
|
|
|
src_fmt->code = format->code;
|
|
src_info = rkisp1_mbus_info_get_by_code(src_fmt->code);
|
|
if (!src_info || !(src_info->direction & RKISP1_ISP_SD_SRC)) {
|
|
src_fmt->code = RKISP1_DEF_SRC_PAD_FMT;
|
|
src_info = rkisp1_mbus_info_get_by_code(src_fmt->code);
|
|
}
|
|
|
|
if (sink_info->pixel_enc == V4L2_PIXEL_ENC_YUV &&
|
|
src_info->pixel_enc == V4L2_PIXEL_ENC_BAYER) {
|
|
src_fmt->code = sink_fmt->code;
|
|
src_info = sink_info;
|
|
}
|
|
|
|
/*
|
|
* The source width and height must be identical to the source crop
|
|
* size.
|
|
*/
|
|
src_fmt->width = src_crop->width;
|
|
src_fmt->height = src_crop->height;
|
|
|
|
/*
|
|
* Copy the color space for the sink pad. When converting from Bayer to
|
|
* YUV, default to a limited quantization range.
|
|
*/
|
|
src_fmt->colorspace = sink_fmt->colorspace;
|
|
src_fmt->xfer_func = sink_fmt->xfer_func;
|
|
src_fmt->ycbcr_enc = sink_fmt->ycbcr_enc;
|
|
|
|
if (sink_info->pixel_enc == V4L2_PIXEL_ENC_BAYER &&
|
|
src_info->pixel_enc == V4L2_PIXEL_ENC_YUV)
|
|
src_fmt->quantization = V4L2_QUANTIZATION_LIM_RANGE;
|
|
else
|
|
src_fmt->quantization = sink_fmt->quantization;
|
|
|
|
/*
|
|
* Allow setting the source color space fields when the SET_CSC flag is
|
|
* set and the source format is YUV. If the sink format is YUV, don't
|
|
* set the color primaries, transfer function or YCbCr encoding as the
|
|
* ISP is bypassed in that case and passes YUV data through without
|
|
* modifications.
|
|
*
|
|
* The color primaries and transfer function are configured through the
|
|
* cross-talk matrix and tone curve respectively. Settings for those
|
|
* hardware blocks are conveyed through the ISP parameters buffer, as
|
|
* they need to combine color space information with other image tuning
|
|
* characteristics and can't thus be computed by the kernel based on the
|
|
* color space. The source pad colorspace and xfer_func fields are thus
|
|
* ignored by the driver, but can be set by userspace to propagate
|
|
* accurate color space information down the pipeline.
|
|
*/
|
|
set_csc = format->flags & V4L2_MBUS_FRAMEFMT_SET_CSC;
|
|
|
|
if (set_csc && src_info->pixel_enc == V4L2_PIXEL_ENC_YUV) {
|
|
if (sink_info->pixel_enc == V4L2_PIXEL_ENC_BAYER) {
|
|
if (format->colorspace != V4L2_COLORSPACE_DEFAULT)
|
|
src_fmt->colorspace = format->colorspace;
|
|
if (format->xfer_func != V4L2_XFER_FUNC_DEFAULT)
|
|
src_fmt->xfer_func = format->xfer_func;
|
|
if (format->ycbcr_enc != V4L2_YCBCR_ENC_DEFAULT)
|
|
src_fmt->ycbcr_enc = format->ycbcr_enc;
|
|
}
|
|
|
|
if (format->quantization != V4L2_QUANTIZATION_DEFAULT)
|
|
src_fmt->quantization = format->quantization;
|
|
}
|
|
|
|
*format = *src_fmt;
|
|
|
|
/*
|
|
* Restore the SET_CSC flag if it was set to indicate support for the
|
|
* CSC setting API.
|
|
*/
|
|
if (set_csc)
|
|
format->flags |= V4L2_MBUS_FRAMEFMT_SET_CSC;
|
|
|
|
/* Store the source format info when setting the active format. */
|
|
if (which == V4L2_SUBDEV_FORMAT_ACTIVE)
|
|
isp->src_fmt = src_info;
|
|
}
|
|
|
|
static void rkisp1_isp_set_src_crop(struct rkisp1_isp *isp,
|
|
struct v4l2_subdev_state *sd_state,
|
|
struct v4l2_rect *r, unsigned int which)
|
|
{
|
|
struct v4l2_mbus_framefmt *src_fmt;
|
|
const struct v4l2_rect *sink_crop;
|
|
struct v4l2_rect *src_crop;
|
|
|
|
src_crop = rkisp1_isp_get_pad_crop(isp, sd_state,
|
|
RKISP1_ISP_PAD_SOURCE_VIDEO,
|
|
which);
|
|
sink_crop = rkisp1_isp_get_pad_crop(isp, sd_state,
|
|
RKISP1_ISP_PAD_SINK_VIDEO,
|
|
which);
|
|
|
|
src_crop->left = ALIGN(r->left, 2);
|
|
src_crop->width = ALIGN(r->width, 2);
|
|
src_crop->top = r->top;
|
|
src_crop->height = r->height;
|
|
rkisp1_sd_adjust_crop_rect(src_crop, sink_crop);
|
|
|
|
*r = *src_crop;
|
|
|
|
/* Propagate to out format */
|
|
src_fmt = rkisp1_isp_get_pad_fmt(isp, sd_state,
|
|
RKISP1_ISP_PAD_SOURCE_VIDEO, which);
|
|
rkisp1_isp_set_src_fmt(isp, sd_state, src_fmt, which);
|
|
}
|
|
|
|
static void rkisp1_isp_set_sink_crop(struct rkisp1_isp *isp,
|
|
struct v4l2_subdev_state *sd_state,
|
|
struct v4l2_rect *r, unsigned int which)
|
|
{
|
|
struct v4l2_rect *sink_crop, *src_crop;
|
|
const struct v4l2_mbus_framefmt *sink_fmt;
|
|
|
|
sink_crop = rkisp1_isp_get_pad_crop(isp, sd_state,
|
|
RKISP1_ISP_PAD_SINK_VIDEO,
|
|
which);
|
|
sink_fmt = rkisp1_isp_get_pad_fmt(isp, sd_state,
|
|
RKISP1_ISP_PAD_SINK_VIDEO,
|
|
which);
|
|
|
|
sink_crop->left = ALIGN(r->left, 2);
|
|
sink_crop->width = ALIGN(r->width, 2);
|
|
sink_crop->top = r->top;
|
|
sink_crop->height = r->height;
|
|
rkisp1_sd_adjust_crop(sink_crop, sink_fmt);
|
|
|
|
*r = *sink_crop;
|
|
|
|
/* Propagate to out crop */
|
|
src_crop = rkisp1_isp_get_pad_crop(isp, sd_state,
|
|
RKISP1_ISP_PAD_SOURCE_VIDEO, which);
|
|
rkisp1_isp_set_src_crop(isp, sd_state, src_crop, which);
|
|
}
|
|
|
|
static void rkisp1_isp_set_sink_fmt(struct rkisp1_isp *isp,
|
|
struct v4l2_subdev_state *sd_state,
|
|
struct v4l2_mbus_framefmt *format,
|
|
unsigned int which)
|
|
{
|
|
const struct rkisp1_mbus_info *mbus_info;
|
|
struct v4l2_mbus_framefmt *sink_fmt;
|
|
struct v4l2_rect *sink_crop;
|
|
bool is_yuv;
|
|
|
|
sink_fmt = rkisp1_isp_get_pad_fmt(isp, sd_state,
|
|
RKISP1_ISP_PAD_SINK_VIDEO,
|
|
which);
|
|
sink_fmt->code = format->code;
|
|
mbus_info = rkisp1_mbus_info_get_by_code(sink_fmt->code);
|
|
if (!mbus_info || !(mbus_info->direction & RKISP1_ISP_SD_SINK)) {
|
|
sink_fmt->code = RKISP1_DEF_SINK_PAD_FMT;
|
|
mbus_info = rkisp1_mbus_info_get_by_code(sink_fmt->code);
|
|
}
|
|
if (which == V4L2_SUBDEV_FORMAT_ACTIVE)
|
|
isp->sink_fmt = mbus_info;
|
|
|
|
sink_fmt->width = clamp_t(u32, format->width,
|
|
RKISP1_ISP_MIN_WIDTH,
|
|
RKISP1_ISP_MAX_WIDTH);
|
|
sink_fmt->height = clamp_t(u32, format->height,
|
|
RKISP1_ISP_MIN_HEIGHT,
|
|
RKISP1_ISP_MAX_HEIGHT);
|
|
|
|
/*
|
|
* Adjust the color space fields. Accept any color primaries and
|
|
* transfer function for both YUV and Bayer. For YUV any YCbCr encoding
|
|
* and quantization range is also accepted. For Bayer formats, the YCbCr
|
|
* encoding isn't applicable, and the quantization range can only be
|
|
* full.
|
|
*/
|
|
is_yuv = mbus_info->pixel_enc == V4L2_PIXEL_ENC_YUV;
|
|
|
|
sink_fmt->colorspace = format->colorspace ? :
|
|
(is_yuv ? V4L2_COLORSPACE_SRGB :
|
|
V4L2_COLORSPACE_RAW);
|
|
sink_fmt->xfer_func = format->xfer_func ? :
|
|
V4L2_MAP_XFER_FUNC_DEFAULT(sink_fmt->colorspace);
|
|
if (is_yuv) {
|
|
sink_fmt->ycbcr_enc = format->ycbcr_enc ? :
|
|
V4L2_MAP_YCBCR_ENC_DEFAULT(sink_fmt->colorspace);
|
|
sink_fmt->quantization = format->quantization ? :
|
|
V4L2_MAP_QUANTIZATION_DEFAULT(false, sink_fmt->colorspace,
|
|
sink_fmt->ycbcr_enc);
|
|
} else {
|
|
/*
|
|
* The YCbCr encoding isn't applicable for non-YUV formats, but
|
|
* V4L2 has no "no encoding" value. Hardcode it to Rec. 601, it
|
|
* should be ignored by userspace.
|
|
*/
|
|
sink_fmt->ycbcr_enc = V4L2_YCBCR_ENC_601;
|
|
sink_fmt->quantization = V4L2_QUANTIZATION_FULL_RANGE;
|
|
}
|
|
|
|
*format = *sink_fmt;
|
|
|
|
/* Propagate to in crop */
|
|
sink_crop = rkisp1_isp_get_pad_crop(isp, sd_state,
|
|
RKISP1_ISP_PAD_SINK_VIDEO,
|
|
which);
|
|
rkisp1_isp_set_sink_crop(isp, sd_state, sink_crop, which);
|
|
}
|
|
|
|
static int rkisp1_isp_get_fmt(struct v4l2_subdev *sd,
|
|
struct v4l2_subdev_state *sd_state,
|
|
struct v4l2_subdev_format *fmt)
|
|
{
|
|
struct rkisp1_isp *isp = to_rkisp1_isp(sd);
|
|
|
|
mutex_lock(&isp->ops_lock);
|
|
fmt->format = *rkisp1_isp_get_pad_fmt(isp, sd_state, fmt->pad,
|
|
fmt->which);
|
|
mutex_unlock(&isp->ops_lock);
|
|
return 0;
|
|
}
|
|
|
|
static int rkisp1_isp_set_fmt(struct v4l2_subdev *sd,
|
|
struct v4l2_subdev_state *sd_state,
|
|
struct v4l2_subdev_format *fmt)
|
|
{
|
|
struct rkisp1_isp *isp = to_rkisp1_isp(sd);
|
|
|
|
mutex_lock(&isp->ops_lock);
|
|
if (fmt->pad == RKISP1_ISP_PAD_SINK_VIDEO)
|
|
rkisp1_isp_set_sink_fmt(isp, sd_state, &fmt->format,
|
|
fmt->which);
|
|
else if (fmt->pad == RKISP1_ISP_PAD_SOURCE_VIDEO)
|
|
rkisp1_isp_set_src_fmt(isp, sd_state, &fmt->format,
|
|
fmt->which);
|
|
else
|
|
fmt->format = *rkisp1_isp_get_pad_fmt(isp, sd_state, fmt->pad,
|
|
fmt->which);
|
|
|
|
mutex_unlock(&isp->ops_lock);
|
|
return 0;
|
|
}
|
|
|
|
static int rkisp1_isp_get_selection(struct v4l2_subdev *sd,
|
|
struct v4l2_subdev_state *sd_state,
|
|
struct v4l2_subdev_selection *sel)
|
|
{
|
|
struct rkisp1_isp *isp = to_rkisp1_isp(sd);
|
|
int ret = 0;
|
|
|
|
if (sel->pad != RKISP1_ISP_PAD_SOURCE_VIDEO &&
|
|
sel->pad != RKISP1_ISP_PAD_SINK_VIDEO)
|
|
return -EINVAL;
|
|
|
|
mutex_lock(&isp->ops_lock);
|
|
switch (sel->target) {
|
|
case V4L2_SEL_TGT_CROP_BOUNDS:
|
|
if (sel->pad == RKISP1_ISP_PAD_SINK_VIDEO) {
|
|
struct v4l2_mbus_framefmt *fmt;
|
|
|
|
fmt = rkisp1_isp_get_pad_fmt(isp, sd_state, sel->pad,
|
|
sel->which);
|
|
sel->r.height = fmt->height;
|
|
sel->r.width = fmt->width;
|
|
sel->r.left = 0;
|
|
sel->r.top = 0;
|
|
} else {
|
|
sel->r = *rkisp1_isp_get_pad_crop(isp, sd_state,
|
|
RKISP1_ISP_PAD_SINK_VIDEO,
|
|
sel->which);
|
|
}
|
|
break;
|
|
case V4L2_SEL_TGT_CROP:
|
|
sel->r = *rkisp1_isp_get_pad_crop(isp, sd_state, sel->pad,
|
|
sel->which);
|
|
break;
|
|
default:
|
|
ret = -EINVAL;
|
|
}
|
|
mutex_unlock(&isp->ops_lock);
|
|
return ret;
|
|
}
|
|
|
|
static int rkisp1_isp_set_selection(struct v4l2_subdev *sd,
|
|
struct v4l2_subdev_state *sd_state,
|
|
struct v4l2_subdev_selection *sel)
|
|
{
|
|
struct rkisp1_isp *isp = to_rkisp1_isp(sd);
|
|
int ret = 0;
|
|
|
|
if (sel->target != V4L2_SEL_TGT_CROP)
|
|
return -EINVAL;
|
|
|
|
dev_dbg(isp->rkisp1->dev, "%s: pad: %d sel(%d,%d)/%dx%d\n", __func__,
|
|
sel->pad, sel->r.left, sel->r.top, sel->r.width, sel->r.height);
|
|
mutex_lock(&isp->ops_lock);
|
|
if (sel->pad == RKISP1_ISP_PAD_SINK_VIDEO)
|
|
rkisp1_isp_set_sink_crop(isp, sd_state, &sel->r, sel->which);
|
|
else if (sel->pad == RKISP1_ISP_PAD_SOURCE_VIDEO)
|
|
rkisp1_isp_set_src_crop(isp, sd_state, &sel->r, sel->which);
|
|
else
|
|
ret = -EINVAL;
|
|
|
|
mutex_unlock(&isp->ops_lock);
|
|
return ret;
|
|
}
|
|
|
|
static int rkisp1_subdev_link_validate(struct media_link *link)
|
|
{
|
|
if (link->sink->index == RKISP1_ISP_PAD_SINK_PARAMS)
|
|
return 0;
|
|
|
|
return v4l2_subdev_link_validate(link);
|
|
}
|
|
|
|
static const struct v4l2_subdev_pad_ops rkisp1_isp_pad_ops = {
|
|
.enum_mbus_code = rkisp1_isp_enum_mbus_code,
|
|
.enum_frame_size = rkisp1_isp_enum_frame_size,
|
|
.get_selection = rkisp1_isp_get_selection,
|
|
.set_selection = rkisp1_isp_set_selection,
|
|
.init_cfg = rkisp1_isp_init_config,
|
|
.get_fmt = rkisp1_isp_get_fmt,
|
|
.set_fmt = rkisp1_isp_set_fmt,
|
|
.link_validate = v4l2_subdev_link_validate_default,
|
|
};
|
|
|
|
/* ----------------------------------------------------------------------------
|
|
* Stream operations
|
|
*/
|
|
|
|
static int rkisp1_isp_s_stream(struct v4l2_subdev *sd, int enable)
|
|
{
|
|
struct rkisp1_isp *isp = to_rkisp1_isp(sd);
|
|
struct rkisp1_device *rkisp1 = isp->rkisp1;
|
|
struct media_pad *source_pad;
|
|
struct media_pad *sink_pad;
|
|
enum v4l2_mbus_type mbus_type;
|
|
u32 mbus_flags;
|
|
int ret;
|
|
|
|
if (!enable) {
|
|
v4l2_subdev_call(rkisp1->source, video, s_stream, false);
|
|
rkisp1_isp_stop(isp);
|
|
return 0;
|
|
}
|
|
|
|
sink_pad = &isp->pads[RKISP1_ISP_PAD_SINK_VIDEO];
|
|
source_pad = media_pad_remote_pad_unique(sink_pad);
|
|
if (IS_ERR(source_pad)) {
|
|
dev_dbg(rkisp1->dev, "Failed to get source for ISP: %ld\n",
|
|
PTR_ERR(source_pad));
|
|
return -EPIPE;
|
|
}
|
|
|
|
rkisp1->source = media_entity_to_v4l2_subdev(source_pad->entity);
|
|
if (!rkisp1->source) {
|
|
/* This should really not happen, so is not worth a message. */
|
|
return -EPIPE;
|
|
}
|
|
|
|
if (rkisp1->source == &rkisp1->csi.sd) {
|
|
mbus_type = V4L2_MBUS_CSI2_DPHY;
|
|
mbus_flags = 0;
|
|
} else {
|
|
const struct rkisp1_sensor_async *asd;
|
|
|
|
asd = container_of(rkisp1->source->asd,
|
|
struct rkisp1_sensor_async, asd);
|
|
|
|
mbus_type = asd->mbus_type;
|
|
mbus_flags = asd->mbus_flags;
|
|
}
|
|
|
|
isp->frame_sequence = -1;
|
|
mutex_lock(&isp->ops_lock);
|
|
ret = rkisp1_config_cif(isp, mbus_type, mbus_flags);
|
|
if (ret)
|
|
goto mutex_unlock;
|
|
|
|
rkisp1_isp_start(isp);
|
|
|
|
ret = v4l2_subdev_call(rkisp1->source, video, s_stream, true);
|
|
if (ret) {
|
|
rkisp1_isp_stop(isp);
|
|
goto mutex_unlock;
|
|
}
|
|
|
|
mutex_unlock:
|
|
mutex_unlock(&isp->ops_lock);
|
|
return ret;
|
|
}
|
|
|
|
static int rkisp1_isp_subs_evt(struct v4l2_subdev *sd, struct v4l2_fh *fh,
|
|
struct v4l2_event_subscription *sub)
|
|
{
|
|
if (sub->type != V4L2_EVENT_FRAME_SYNC)
|
|
return -EINVAL;
|
|
|
|
/* V4L2_EVENT_FRAME_SYNC doesn't require an id, so zero should be set */
|
|
if (sub->id != 0)
|
|
return -EINVAL;
|
|
|
|
return v4l2_event_subscribe(fh, sub, 0, NULL);
|
|
}
|
|
|
|
static const struct media_entity_operations rkisp1_isp_media_ops = {
|
|
.link_validate = rkisp1_subdev_link_validate,
|
|
};
|
|
|
|
static const struct v4l2_subdev_video_ops rkisp1_isp_video_ops = {
|
|
.s_stream = rkisp1_isp_s_stream,
|
|
};
|
|
|
|
static const struct v4l2_subdev_core_ops rkisp1_isp_core_ops = {
|
|
.subscribe_event = rkisp1_isp_subs_evt,
|
|
.unsubscribe_event = v4l2_event_subdev_unsubscribe,
|
|
};
|
|
|
|
static const struct v4l2_subdev_ops rkisp1_isp_ops = {
|
|
.core = &rkisp1_isp_core_ops,
|
|
.video = &rkisp1_isp_video_ops,
|
|
.pad = &rkisp1_isp_pad_ops,
|
|
};
|
|
|
|
int rkisp1_isp_register(struct rkisp1_device *rkisp1)
|
|
{
|
|
struct v4l2_subdev_state state = {
|
|
.pads = rkisp1->isp.pad_cfg
|
|
};
|
|
struct rkisp1_isp *isp = &rkisp1->isp;
|
|
struct media_pad *pads = isp->pads;
|
|
struct v4l2_subdev *sd = &isp->sd;
|
|
int ret;
|
|
|
|
isp->rkisp1 = rkisp1;
|
|
|
|
v4l2_subdev_init(sd, &rkisp1_isp_ops);
|
|
sd->flags |= V4L2_SUBDEV_FL_HAS_DEVNODE | V4L2_SUBDEV_FL_HAS_EVENTS;
|
|
sd->entity.ops = &rkisp1_isp_media_ops;
|
|
sd->entity.function = MEDIA_ENT_F_PROC_VIDEO_PIXEL_FORMATTER;
|
|
sd->owner = THIS_MODULE;
|
|
strscpy(sd->name, RKISP1_ISP_DEV_NAME, sizeof(sd->name));
|
|
|
|
pads[RKISP1_ISP_PAD_SINK_VIDEO].flags = MEDIA_PAD_FL_SINK |
|
|
MEDIA_PAD_FL_MUST_CONNECT;
|
|
pads[RKISP1_ISP_PAD_SINK_PARAMS].flags = MEDIA_PAD_FL_SINK;
|
|
pads[RKISP1_ISP_PAD_SOURCE_VIDEO].flags = MEDIA_PAD_FL_SOURCE;
|
|
pads[RKISP1_ISP_PAD_SOURCE_STATS].flags = MEDIA_PAD_FL_SOURCE;
|
|
|
|
isp->sink_fmt = rkisp1_mbus_info_get_by_code(RKISP1_DEF_SINK_PAD_FMT);
|
|
isp->src_fmt = rkisp1_mbus_info_get_by_code(RKISP1_DEF_SRC_PAD_FMT);
|
|
|
|
mutex_init(&isp->ops_lock);
|
|
ret = media_entity_pads_init(&sd->entity, RKISP1_ISP_PAD_MAX, pads);
|
|
if (ret)
|
|
goto error;
|
|
|
|
ret = v4l2_device_register_subdev(&rkisp1->v4l2_dev, sd);
|
|
if (ret) {
|
|
dev_err(rkisp1->dev, "Failed to register isp subdev\n");
|
|
goto error;
|
|
}
|
|
|
|
rkisp1_isp_init_config(sd, &state);
|
|
|
|
return 0;
|
|
|
|
error:
|
|
media_entity_cleanup(&sd->entity);
|
|
mutex_destroy(&isp->ops_lock);
|
|
isp->sd.v4l2_dev = NULL;
|
|
return ret;
|
|
}
|
|
|
|
void rkisp1_isp_unregister(struct rkisp1_device *rkisp1)
|
|
{
|
|
struct rkisp1_isp *isp = &rkisp1->isp;
|
|
|
|
if (!isp->sd.v4l2_dev)
|
|
return;
|
|
|
|
v4l2_device_unregister_subdev(&isp->sd);
|
|
media_entity_cleanup(&isp->sd.entity);
|
|
mutex_destroy(&isp->ops_lock);
|
|
}
|
|
|
|
/* ----------------------------------------------------------------------------
|
|
* Interrupt handlers
|
|
*/
|
|
|
|
static void rkisp1_isp_queue_event_sof(struct rkisp1_isp *isp)
|
|
{
|
|
struct v4l2_event event = {
|
|
.type = V4L2_EVENT_FRAME_SYNC,
|
|
};
|
|
|
|
event.u.frame_sync.frame_sequence = isp->frame_sequence;
|
|
v4l2_event_queue(isp->sd.devnode, &event);
|
|
}
|
|
|
|
irqreturn_t rkisp1_isp_isr(int irq, void *ctx)
|
|
{
|
|
struct device *dev = ctx;
|
|
struct rkisp1_device *rkisp1 = dev_get_drvdata(dev);
|
|
u32 status, isp_err;
|
|
|
|
status = rkisp1_read(rkisp1, RKISP1_CIF_ISP_MIS);
|
|
if (!status)
|
|
return IRQ_NONE;
|
|
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_ICR, status);
|
|
|
|
/* Vertical sync signal, starting generating new frame */
|
|
if (status & RKISP1_CIF_ISP_V_START) {
|
|
rkisp1->isp.frame_sequence++;
|
|
rkisp1_isp_queue_event_sof(&rkisp1->isp);
|
|
if (status & RKISP1_CIF_ISP_FRAME) {
|
|
WARN_ONCE(1, "irq delay is too long, buffers might not be in sync\n");
|
|
rkisp1->debug.irq_delay++;
|
|
}
|
|
}
|
|
if (status & RKISP1_CIF_ISP_PIC_SIZE_ERROR) {
|
|
/* Clear pic_size_error */
|
|
isp_err = rkisp1_read(rkisp1, RKISP1_CIF_ISP_ERR);
|
|
if (isp_err & RKISP1_CIF_ISP_ERR_INFORM_SIZE)
|
|
rkisp1->debug.inform_size_error++;
|
|
if (isp_err & RKISP1_CIF_ISP_ERR_IS_SIZE)
|
|
rkisp1->debug.img_stabilization_size_error++;
|
|
if (isp_err & RKISP1_CIF_ISP_ERR_OUTFORM_SIZE)
|
|
rkisp1->debug.outform_size_error++;
|
|
rkisp1_write(rkisp1, RKISP1_CIF_ISP_ERR_CLR, isp_err);
|
|
} else if (status & RKISP1_CIF_ISP_DATA_LOSS) {
|
|
/* keep track of data_loss in debugfs */
|
|
rkisp1->debug.data_loss++;
|
|
}
|
|
|
|
if (status & RKISP1_CIF_ISP_FRAME) {
|
|
u32 isp_ris;
|
|
|
|
/* New frame from the sensor received */
|
|
isp_ris = rkisp1_read(rkisp1, RKISP1_CIF_ISP_RIS);
|
|
if (isp_ris & RKISP1_STATS_MEAS_MASK)
|
|
rkisp1_stats_isr(&rkisp1->stats, isp_ris);
|
|
/*
|
|
* Then update changed configs. Some of them involve
|
|
* lot of register writes. Do those only one per frame.
|
|
* Do the updates in the order of the processing flow.
|
|
*/
|
|
rkisp1_params_isr(rkisp1);
|
|
}
|
|
|
|
return IRQ_HANDLED;
|
|
}
|