media: imx335: Support vertical flip

Support vertical flip by setting REG_VREVERSE.
Additional registers also needs to be set per mode, according
to the readout direction (normal/inverted) as mentioned in the
data sheet.

Since the register IMX335_REG_AREA3_ST_ADR_1 is based on the
flip (and is set via vflip related registers), it has been
moved out of the 2592x1944 mode regs.

Signed-off-by: Umang Jain <umang.jain@ideasonboard.com>
Reviewed-by: Tommaso Merciai <tomm.merciai@gmail.com>
Reviewed-by: Kieran Bingham <kieran.bingham@ideasonboard.com>
Signed-off-by: Jai Luthra <jai.luthra@ideasonboard.com>
Signed-off-by: Sakari Ailus <sakari.ailus@linux.intel.com>
Signed-off-by: Hans Verkuil <hverkuil+cisco@kernel.org>
This commit is contained in:
Umang Jain 2025-10-30 14:12:55 +05:30 committed by Hans Verkuil
parent 554f7b8d45
commit 2581a40115

View File

@ -56,6 +56,9 @@
#define IMX335_AGAIN_STEP 1
#define IMX335_AGAIN_DEFAULT 0
/* Vertical flip */
#define IMX335_REG_VREVERSE CCI_REG8(0x304f)
#define IMX335_REG_TPG_TESTCLKEN CCI_REG8(0x3148)
#define IMX335_REG_INCLKSEL1 CCI_REG16_LE(0x314c)
@ -155,6 +158,8 @@ static const char * const imx335_supply_name[] = {
* @vblank_max: Maximum vertical blanking in lines
* @pclk: Sensor pixel clock
* @reg_list: Register list for sensor mode
* @vflip_normal: Register list vflip (normal readout)
* @vflip_inverted: Register list vflip (inverted readout)
*/
struct imx335_mode {
u32 width;
@ -166,6 +171,8 @@ struct imx335_mode {
u32 vblank_max;
u64 pclk;
struct imx335_reg_list reg_list;
struct imx335_reg_list vflip_normal;
struct imx335_reg_list vflip_inverted;
};
/**
@ -183,6 +190,7 @@ struct imx335_mode {
* @pclk_ctrl: Pointer to pixel clock control
* @hblank_ctrl: Pointer to horizontal blanking control
* @vblank_ctrl: Pointer to vertical blanking control
* @vflip: Pointer to vertical flip control
* @exp_ctrl: Pointer to exposure control
* @again_ctrl: Pointer to analog gain control
* @vblank: Vertical blanking in lines
@ -207,6 +215,7 @@ struct imx335 {
struct v4l2_ctrl *pclk_ctrl;
struct v4l2_ctrl *hblank_ctrl;
struct v4l2_ctrl *vblank_ctrl;
struct v4l2_ctrl *vflip;
struct {
struct v4l2_ctrl *exp_ctrl;
struct v4l2_ctrl *again_ctrl;
@ -259,7 +268,6 @@ static const struct cci_reg_sequence mode_2592x1944_regs[] = {
{ IMX335_REG_HTRIMMING_START, 48 },
{ IMX335_REG_HNUM, 2592 },
{ IMX335_REG_Y_OUT_SIZE, 1944 },
{ IMX335_REG_AREA3_ST_ADR_1, 176 },
{ IMX335_REG_AREA3_WIDTH_1, 3928 },
{ IMX335_REG_OPB_SIZE_V, 0 },
{ IMX335_REG_XVS_XHS_DRV, 0x00 },
@ -333,6 +341,26 @@ static const struct cci_reg_sequence mode_2592x1944_regs[] = {
{ CCI_REG8(0x3a00), 0x00 },
};
static const struct cci_reg_sequence mode_2592x1944_vflip_normal[] = {
{ IMX335_REG_AREA3_ST_ADR_1, 176 },
/* Undocumented V-Flip related registers on Page 55 of datasheet. */
{ CCI_REG8(0x3081), 0x02, },
{ CCI_REG8(0x3083), 0x02, },
{ CCI_REG16_LE(0x30b6), 0x00 },
{ CCI_REG16_LE(0x3116), 0x08 },
};
static const struct cci_reg_sequence mode_2592x1944_vflip_inverted[] = {
{ IMX335_REG_AREA3_ST_ADR_1, 4112 },
/* Undocumented V-Flip related registers on Page 55 of datasheet. */
{ CCI_REG8(0x3081), 0xfe, },
{ CCI_REG8(0x3083), 0xfe, },
{ CCI_REG16_LE(0x30b6), 0x1fa },
{ CCI_REG16_LE(0x3116), 0x002 },
};
static const struct cci_reg_sequence raw10_framefmt_regs[] = {
{ IMX335_REG_ADBIT, 0x00 },
{ IMX335_REG_MDBIT, 0x00 },
@ -419,6 +447,14 @@ static const struct imx335_mode supported_mode = {
.num_of_regs = ARRAY_SIZE(mode_2592x1944_regs),
.regs = mode_2592x1944_regs,
},
.vflip_normal = {
.num_of_regs = ARRAY_SIZE(mode_2592x1944_vflip_normal),
.regs = mode_2592x1944_vflip_normal,
},
.vflip_inverted = {
.num_of_regs = ARRAY_SIZE(mode_2592x1944_vflip_inverted),
.regs = mode_2592x1944_vflip_inverted,
},
};
/**
@ -492,6 +528,19 @@ static int imx335_update_exp_gain(struct imx335 *imx335, u32 exposure, u32 gain)
return ret;
}
static int imx335_update_vertical_flip(struct imx335 *imx335, u32 vflip)
{
const struct imx335_reg_list * const vflip_regs =
vflip ? &imx335->cur_mode->vflip_inverted :
&imx335->cur_mode->vflip_normal;
int ret = 0;
cci_multi_reg_write(imx335->cci, vflip_regs->regs,
vflip_regs->num_of_regs, &ret);
return cci_write(imx335->cci, IMX335_REG_VREVERSE, vflip, &ret);
}
static int imx335_update_test_pattern(struct imx335 *imx335, u32 pattern_index)
{
int ret = 0;
@ -593,6 +642,10 @@ static int imx335_set_ctrl(struct v4l2_ctrl *ctrl)
ret = imx335_update_exp_gain(imx335, exposure, analog_gain);
break;
case V4L2_CID_VFLIP:
ret = imx335_update_vertical_flip(imx335, ctrl->val);
break;
case V4L2_CID_TEST_PATTERN:
ret = imx335_update_test_pattern(imx335, ctrl->val);
@ -1175,7 +1228,7 @@ static int imx335_init_controls(struct imx335 *imx335)
return ret;
/* v4l2_fwnode_device_properties can add two more controls */
ret = v4l2_ctrl_handler_init(ctrl_hdlr, 9);
ret = v4l2_ctrl_handler_init(ctrl_hdlr, 10);
if (ret)
return ret;
@ -1210,6 +1263,13 @@ static int imx335_init_controls(struct imx335 *imx335)
v4l2_ctrl_cluster(2, &imx335->exp_ctrl);
imx335->vflip = v4l2_ctrl_new_std(ctrl_hdlr,
&imx335_ctrl_ops,
V4L2_CID_VFLIP,
0, 1, 1, 0);
if (imx335->vflip)
imx335->vflip->flags |= V4L2_CTRL_FLAG_MODIFY_LAYOUT;
imx335->vblank_ctrl = v4l2_ctrl_new_std(ctrl_hdlr,
&imx335_ctrl_ops,
V4L2_CID_VBLANK,