Commit f1363166 authored by Hsin-Yi Wang's avatar Hsin-Yi Wang Committed by Mauro Carvalho Chehab

media: ov8856: Set default mbus format but allow caller to alter

Setting the value of V_WIN_OFF (0x3818) from 0x02 to 0x01 to use GRBG
format still results in wrong color output if data is tuned in BGGR mode
before.

Set default mbus format for the supported modes, but allow the caller of
set(get)_fmt to change the bayer format between BGGR and GRBG.

Set the default mbus format for 3264x2448 (and 1632x1224) to BGGR as the
data sheet states the value of this reg should be 0x02 by default.

If new modes are added in the future, they can add the
mipi_data_mbus_{format} settings into bayer_offset_configs to adjust their
offset regs.

Fixes: 2984b0dd ("media: ov8856: Configure sensor for GRBG Bayer for all modes")
Signed-off-by: default avatarHsin-Yi Wang <hsinyi@chromium.org>
Reviewed-by: default avatarRobert Foss <robert.foss@linaro.org>
Signed-off-by: default avatarSakari Ailus <sakari.ailus@linux.intel.com>
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab+huawei@kernel.org>
parent 7ee85054
...@@ -107,6 +107,11 @@ static const char * const ov8856_supply_names[] = { ...@@ -107,6 +107,11 @@ static const char * const ov8856_supply_names[] = {
"dvdd", /* Digital core power */ "dvdd", /* Digital core power */
}; };
enum {
OV8856_MEDIA_BUS_FMT_SBGGR10_1X10,
OV8856_MEDIA_BUS_FMT_SGRBG10_1X10,
};
struct ov8856_reg { struct ov8856_reg {
u16 address; u16 address;
u8 val; u8 val;
...@@ -145,6 +150,9 @@ struct ov8856_mode { ...@@ -145,6 +150,9 @@ struct ov8856_mode {
/* Number of data lanes */ /* Number of data lanes */
u8 data_lanes; u8 data_lanes;
/* Default MEDIA_BUS_FMT for this mode */
u32 default_mbus_index;
}; };
struct ov8856_mipi_data_rates { struct ov8856_mipi_data_rates {
...@@ -1055,7 +1063,7 @@ static const struct ov8856_reg lane_4_mode_3264x2448[] = { ...@@ -1055,7 +1063,7 @@ static const struct ov8856_reg lane_4_mode_3264x2448[] = {
{0x3810, 0x00}, {0x3810, 0x00},
{0x3811, 0x04}, {0x3811, 0x04},
{0x3812, 0x00}, {0x3812, 0x00},
{0x3813, 0x01}, {0x3813, 0x02},
{0x3814, 0x01}, {0x3814, 0x01},
{0x3815, 0x01}, {0x3815, 0x01},
{0x3816, 0x00}, {0x3816, 0x00},
...@@ -1259,7 +1267,7 @@ static const struct ov8856_reg lane_4_mode_1632x1224[] = { ...@@ -1259,7 +1267,7 @@ static const struct ov8856_reg lane_4_mode_1632x1224[] = {
{0x3810, 0x00}, {0x3810, 0x00},
{0x3811, 0x02}, {0x3811, 0x02},
{0x3812, 0x00}, {0x3812, 0x00},
{0x3813, 0x01}, {0x3813, 0x02},
{0x3814, 0x03}, {0x3814, 0x03},
{0x3815, 0x01}, {0x3815, 0x01},
{0x3816, 0x00}, {0x3816, 0x00},
...@@ -1372,6 +1380,19 @@ static const struct ov8856_reg lane_4_mode_1632x1224[] = { ...@@ -1372,6 +1380,19 @@ static const struct ov8856_reg lane_4_mode_1632x1224[] = {
{0x5e10, 0xfc} {0x5e10, 0xfc}
}; };
static const struct ov8856_reg mipi_data_mbus_sbggr10_1x10[] = {
{0x3813, 0x02},
};
static const struct ov8856_reg mipi_data_mbus_sgrbg10_1x10[] = {
{0x3813, 0x01},
};
static const u32 ov8856_mbus_codes[] = {
MEDIA_BUS_FMT_SBGGR10_1X10,
MEDIA_BUS_FMT_SGRBG10_1X10
};
static const char * const ov8856_test_pattern_menu[] = { static const char * const ov8856_test_pattern_menu[] = {
"Disabled", "Disabled",
"Standard Color Bar", "Standard Color Bar",
...@@ -1380,6 +1401,17 @@ static const char * const ov8856_test_pattern_menu[] = { ...@@ -1380,6 +1401,17 @@ static const char * const ov8856_test_pattern_menu[] = {
"Bottom-Top Darker Color Bar" "Bottom-Top Darker Color Bar"
}; };
static const struct ov8856_reg_list bayer_offset_configs[] = {
[OV8856_MEDIA_BUS_FMT_SBGGR10_1X10] = {
.num_of_regs = ARRAY_SIZE(mipi_data_mbus_sbggr10_1x10),
.regs = mipi_data_mbus_sbggr10_1x10,
},
[OV8856_MEDIA_BUS_FMT_SGRBG10_1X10] = {
.num_of_regs = ARRAY_SIZE(mipi_data_mbus_sgrbg10_1x10),
.regs = mipi_data_mbus_sgrbg10_1x10,
}
};
struct ov8856 { struct ov8856 {
struct v4l2_subdev sd; struct v4l2_subdev sd;
struct media_pad pad; struct media_pad pad;
...@@ -1399,6 +1431,9 @@ struct ov8856 { ...@@ -1399,6 +1431,9 @@ struct ov8856 {
/* Current mode */ /* Current mode */
const struct ov8856_mode *cur_mode; const struct ov8856_mode *cur_mode;
/* Application specified mbus format */
u32 cur_mbus_index;
/* To serialize asynchronus callbacks */ /* To serialize asynchronus callbacks */
struct mutex mutex; struct mutex mutex;
...@@ -1450,6 +1485,7 @@ static const struct ov8856_lane_cfg lane_cfg_2 = { ...@@ -1450,6 +1485,7 @@ static const struct ov8856_lane_cfg lane_cfg_2 = {
}, },
.link_freq_index = 0, .link_freq_index = 0,
.data_lanes = 2, .data_lanes = 2,
.default_mbus_index = OV8856_MEDIA_BUS_FMT_SGRBG10_1X10,
}, },
{ {
.width = 1640, .width = 1640,
...@@ -1464,6 +1500,7 @@ static const struct ov8856_lane_cfg lane_cfg_2 = { ...@@ -1464,6 +1500,7 @@ static const struct ov8856_lane_cfg lane_cfg_2 = {
}, },
.link_freq_index = 1, .link_freq_index = 1,
.data_lanes = 2, .data_lanes = 2,
.default_mbus_index = OV8856_MEDIA_BUS_FMT_SGRBG10_1X10,
}} }}
}; };
...@@ -1499,6 +1536,7 @@ static const struct ov8856_lane_cfg lane_cfg_4 = { ...@@ -1499,6 +1536,7 @@ static const struct ov8856_lane_cfg lane_cfg_4 = {
}, },
.link_freq_index = 0, .link_freq_index = 0,
.data_lanes = 4, .data_lanes = 4,
.default_mbus_index = OV8856_MEDIA_BUS_FMT_SGRBG10_1X10,
}, },
{ {
.width = 1640, .width = 1640,
...@@ -1513,6 +1551,7 @@ static const struct ov8856_lane_cfg lane_cfg_4 = { ...@@ -1513,6 +1551,7 @@ static const struct ov8856_lane_cfg lane_cfg_4 = {
}, },
.link_freq_index = 1, .link_freq_index = 1,
.data_lanes = 4, .data_lanes = 4,
.default_mbus_index = OV8856_MEDIA_BUS_FMT_SGRBG10_1X10,
}, },
{ {
.width = 3264, .width = 3264,
...@@ -1527,6 +1566,7 @@ static const struct ov8856_lane_cfg lane_cfg_4 = { ...@@ -1527,6 +1566,7 @@ static const struct ov8856_lane_cfg lane_cfg_4 = {
}, },
.link_freq_index = 0, .link_freq_index = 0,
.data_lanes = 4, .data_lanes = 4,
.default_mbus_index = OV8856_MEDIA_BUS_FMT_SBGGR10_1X10,
}, },
{ {
.width = 1632, .width = 1632,
...@@ -1541,6 +1581,7 @@ static const struct ov8856_lane_cfg lane_cfg_4 = { ...@@ -1541,6 +1581,7 @@ static const struct ov8856_lane_cfg lane_cfg_4 = {
}, },
.link_freq_index = 1, .link_freq_index = 1,
.data_lanes = 4, .data_lanes = 4,
.default_mbus_index = OV8856_MEDIA_BUS_FMT_SBGGR10_1X10,
}} }}
}; };
...@@ -1904,12 +1945,21 @@ static int ov8856_init_controls(struct ov8856 *ov8856) ...@@ -1904,12 +1945,21 @@ static int ov8856_init_controls(struct ov8856 *ov8856)
return 0; return 0;
} }
static void ov8856_update_pad_format(const struct ov8856_mode *mode, static void ov8856_update_pad_format(struct ov8856 *ov8856,
const struct ov8856_mode *mode,
struct v4l2_mbus_framefmt *fmt) struct v4l2_mbus_framefmt *fmt)
{ {
int index;
fmt->width = mode->width; fmt->width = mode->width;
fmt->height = mode->height; fmt->height = mode->height;
fmt->code = MEDIA_BUS_FMT_SGRBG10_1X10; for (index = 0; index < ARRAY_SIZE(ov8856_mbus_codes); ++index)
if (ov8856_mbus_codes[index] == fmt->code)
break;
if (index == ARRAY_SIZE(ov8856_mbus_codes))
index = mode->default_mbus_index;
fmt->code = ov8856_mbus_codes[index];
ov8856->cur_mbus_index = index;
fmt->field = V4L2_FIELD_NONE; fmt->field = V4L2_FIELD_NONE;
} }
...@@ -1935,6 +1985,13 @@ static int ov8856_start_streaming(struct ov8856 *ov8856) ...@@ -1935,6 +1985,13 @@ static int ov8856_start_streaming(struct ov8856 *ov8856)
return ret; return ret;
} }
reg_list = &bayer_offset_configs[ov8856->cur_mbus_index];
ret = ov8856_write_reg_list(ov8856, reg_list);
if (ret) {
dev_err(&client->dev, "failed to set mbus format");
return ret;
}
ret = __v4l2_ctrl_handler_setup(ov8856->sd.ctrl_handler); ret = __v4l2_ctrl_handler_setup(ov8856->sd.ctrl_handler);
if (ret) if (ret)
return ret; return ret;
...@@ -2096,7 +2153,7 @@ static int ov8856_set_format(struct v4l2_subdev *sd, ...@@ -2096,7 +2153,7 @@ static int ov8856_set_format(struct v4l2_subdev *sd,
fmt->format.height); fmt->format.height);
mutex_lock(&ov8856->mutex); mutex_lock(&ov8856->mutex);
ov8856_update_pad_format(mode, &fmt->format); ov8856_update_pad_format(ov8856, mode, &fmt->format);
if (fmt->which == V4L2_SUBDEV_FORMAT_TRY) { if (fmt->which == V4L2_SUBDEV_FORMAT_TRY) {
*v4l2_subdev_get_try_format(sd, sd_state, fmt->pad) = fmt->format; *v4l2_subdev_get_try_format(sd, sd_state, fmt->pad) = fmt->format;
} else { } else {
...@@ -2140,7 +2197,7 @@ static int ov8856_get_format(struct v4l2_subdev *sd, ...@@ -2140,7 +2197,7 @@ static int ov8856_get_format(struct v4l2_subdev *sd,
sd_state, sd_state,
fmt->pad); fmt->pad);
else else
ov8856_update_pad_format(ov8856->cur_mode, &fmt->format); ov8856_update_pad_format(ov8856, ov8856->cur_mode, &fmt->format);
mutex_unlock(&ov8856->mutex); mutex_unlock(&ov8856->mutex);
...@@ -2151,11 +2208,10 @@ static int ov8856_enum_mbus_code(struct v4l2_subdev *sd, ...@@ -2151,11 +2208,10 @@ static int ov8856_enum_mbus_code(struct v4l2_subdev *sd,
struct v4l2_subdev_state *sd_state, struct v4l2_subdev_state *sd_state,
struct v4l2_subdev_mbus_code_enum *code) struct v4l2_subdev_mbus_code_enum *code)
{ {
/* Only one bayer order GRBG is supported */ if (code->index >= ARRAY_SIZE(ov8856_mbus_codes))
if (code->index > 0)
return -EINVAL; return -EINVAL;
code->code = MEDIA_BUS_FMT_SGRBG10_1X10; code->code = ov8856_mbus_codes[code->index];
return 0; return 0;
} }
...@@ -2165,11 +2221,15 @@ static int ov8856_enum_frame_size(struct v4l2_subdev *sd, ...@@ -2165,11 +2221,15 @@ static int ov8856_enum_frame_size(struct v4l2_subdev *sd,
struct v4l2_subdev_frame_size_enum *fse) struct v4l2_subdev_frame_size_enum *fse)
{ {
struct ov8856 *ov8856 = to_ov8856(sd); struct ov8856 *ov8856 = to_ov8856(sd);
int index;
if (fse->index >= ov8856->modes_size) if (fse->index >= ov8856->modes_size)
return -EINVAL; return -EINVAL;
if (fse->code != MEDIA_BUS_FMT_SGRBG10_1X10) for (index = 0; index < ARRAY_SIZE(ov8856_mbus_codes); ++index)
if (fse->code == ov8856_mbus_codes[index])
break;
if (index == ARRAY_SIZE(ov8856_mbus_codes))
return -EINVAL; return -EINVAL;
fse->min_width = ov8856->priv_lane->supported_modes[fse->index].width; fse->min_width = ov8856->priv_lane->supported_modes[fse->index].width;
...@@ -2185,7 +2245,7 @@ static int ov8856_open(struct v4l2_subdev *sd, struct v4l2_subdev_fh *fh) ...@@ -2185,7 +2245,7 @@ static int ov8856_open(struct v4l2_subdev *sd, struct v4l2_subdev_fh *fh)
struct ov8856 *ov8856 = to_ov8856(sd); struct ov8856 *ov8856 = to_ov8856(sd);
mutex_lock(&ov8856->mutex); mutex_lock(&ov8856->mutex);
ov8856_update_pad_format(&ov8856->priv_lane->supported_modes[0], ov8856_update_pad_format(ov8856, &ov8856->priv_lane->supported_modes[0],
v4l2_subdev_get_try_format(sd, fh->state, 0)); v4l2_subdev_get_try_format(sd, fh->state, 0));
mutex_unlock(&ov8856->mutex); mutex_unlock(&ov8856->mutex);
...@@ -2426,6 +2486,7 @@ static int ov8856_probe(struct i2c_client *client) ...@@ -2426,6 +2486,7 @@ static int ov8856_probe(struct i2c_client *client)
mutex_init(&ov8856->mutex); mutex_init(&ov8856->mutex);
ov8856->cur_mode = &ov8856->priv_lane->supported_modes[0]; ov8856->cur_mode = &ov8856->priv_lane->supported_modes[0];
ov8856->cur_mbus_index = ov8856->cur_mode->default_mbus_index;
ret = ov8856_init_controls(ov8856); ret = ov8856_init_controls(ov8856);
if (ret) { if (ret) {
dev_err(&client->dev, "failed to init controls: %d", ret); dev_err(&client->dev, "failed to init controls: %d", ret);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment