media: rkisp1: Configure CSM based on YCbCr encoding

The driver currently only implements the Rec. 601 YCbCr encoding, extend
it with support for the other encodings defined by V4L2 (Rec. 709, Rec.
2020 and SMPTE240m). The coefficients have been calculated by rounding
the floating point values to the nearest Q1.7 fixed-point value,
adjusting the rounding to ensure that the sum of each line in the matrix
is preserved to avoid overflows.

At the hardware level, the RGB to YUV conversion matrix is fully
configurable, custom encoding could be supported by extending the ISP
parameters if desired.

Signed-off-by: Laurent Pinchart <laurent.pinchart@ideasonboard.com>
Reviewed-by: Dafna Hirschfeld <dafna@fastmail.com>
Reviewed-by: Paul Elder <paul.elder@ideasonboard.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@kernel.org>
This commit is contained in:
Laurent Pinchart 2022-08-13 00:44:14 +02:00 committed by Mauro Carvalho Chehab
parent 711d91497e
commit f7aa2d2343
3 changed files with 84 additions and 21 deletions

View File

@ -378,6 +378,7 @@ struct rkisp1_params {
struct v4l2_format vdev_fmt; struct v4l2_format vdev_fmt;
enum v4l2_quantization quantization; enum v4l2_quantization quantization;
enum v4l2_ycbcr_encoding ycbcr_encoding;
enum rkisp1_fmt_raw_pat_type raw_type; enum rkisp1_fmt_raw_pat_type raw_type;
}; };
@ -563,10 +564,12 @@ const struct rkisp1_mbus_info *rkisp1_mbus_info_get_by_code(u32 mbus_code);
* @params: pointer to rkisp1_params. * @params: pointer to rkisp1_params.
* @bayer_pat: the bayer pattern on the isp video sink pad * @bayer_pat: the bayer pattern on the isp video sink pad
* @quantization: the quantization configured on the isp's src pad * @quantization: the quantization configured on the isp's src pad
* @ycbcr_encoding: the ycbcr_encoding configured on the isp's src pad
*/ */
void rkisp1_params_configure(struct rkisp1_params *params, void rkisp1_params_configure(struct rkisp1_params *params,
enum rkisp1_fmt_raw_pat_type bayer_pat, enum rkisp1_fmt_raw_pat_type bayer_pat,
enum v4l2_quantization quantization); enum v4l2_quantization quantization,
enum v4l2_ycbcr_encoding ycbcr_encoding);
/* rkisp1_params_disable - disable all parameters. /* rkisp1_params_disable - disable all parameters.
* This function is called by the isp entity upon stream start * This function is called by the isp entity upon stream start

View File

@ -234,7 +234,8 @@ static int rkisp1_config_isp(struct rkisp1_isp *isp,
RKISP1_ISP_PAD_SOURCE_VIDEO, RKISP1_ISP_PAD_SOURCE_VIDEO,
V4L2_SUBDEV_FORMAT_ACTIVE); V4L2_SUBDEV_FORMAT_ACTIVE);
rkisp1_params_configure(&rkisp1->params, sink_fmt->bayer_pat, rkisp1_params_configure(&rkisp1->params, sink_fmt->bayer_pat,
src_frm->quantization); src_frm->quantization,
src_frm->ycbcr_enc);
} }
return 0; return 0;

View File

@ -1078,37 +1078,94 @@ static void rkisp1_ie_enable(struct rkisp1_params *params, bool en)
static void rkisp1_csm_config(struct rkisp1_params *params) static void rkisp1_csm_config(struct rkisp1_params *params)
{ {
static const u16 full_range_coeff[] = { struct csm_coeffs {
0x0026, 0x004b, 0x000f, u16 limited[9];
0x01ea, 0x01d6, 0x0040, u16 full[9];
0x0040, 0x01ca, 0x01f6
}; };
static const u16 limited_range_coeff[] = { static const struct csm_coeffs rec601_coeffs = {
0x0021, 0x0040, 0x000d, .limited = {
0x01ed, 0x01db, 0x0038, 0x0021, 0x0042, 0x000d,
0x0038, 0x01d1, 0x01f7, 0x01ed, 0x01db, 0x0038,
0x0038, 0x01d1, 0x01f7,
},
.full = {
0x0026, 0x004b, 0x000f,
0x01ea, 0x01d6, 0x0040,
0x0040, 0x01ca, 0x01f6,
},
}; };
static const struct csm_coeffs rec709_coeffs = {
.limited = {
0x0018, 0x0050, 0x0008,
0x01f3, 0x01d5, 0x0038,
0x0038, 0x01cd, 0x01fb,
},
.full = {
0x001b, 0x005c, 0x0009,
0x01f1, 0x01cf, 0x0040,
0x0040, 0x01c6, 0x01fa,
},
};
static const struct csm_coeffs rec2020_coeffs = {
.limited = {
0x001d, 0x004c, 0x0007,
0x01f0, 0x01d8, 0x0038,
0x0038, 0x01cd, 0x01fb,
},
.full = {
0x0022, 0x0057, 0x0008,
0x01ee, 0x01d2, 0x0040,
0x0040, 0x01c5, 0x01fb,
},
};
static const struct csm_coeffs smpte240m_coeffs = {
.limited = {
0x0018, 0x004f, 0x000a,
0x01f3, 0x01d5, 0x0038,
0x0038, 0x01ce, 0x01fa,
},
.full = {
0x001b, 0x005a, 0x000b,
0x01f1, 0x01cf, 0x0040,
0x0040, 0x01c7, 0x01f9,
},
};
const struct csm_coeffs *coeffs;
const u16 *csm;
unsigned int i; unsigned int i;
if (params->quantization == V4L2_QUANTIZATION_FULL_RANGE) { switch (params->ycbcr_encoding) {
for (i = 0; i < ARRAY_SIZE(full_range_coeff); i++) case V4L2_YCBCR_ENC_601:
rkisp1_write(params->rkisp1, default:
RKISP1_CIF_ISP_CC_COEFF_0 + i * 4, coeffs = &rec601_coeffs;
full_range_coeff[i]); break;
case V4L2_YCBCR_ENC_709:
coeffs = &rec709_coeffs;
break;
case V4L2_YCBCR_ENC_BT2020:
coeffs = &rec2020_coeffs;
break;
case V4L2_YCBCR_ENC_SMPTE240M:
coeffs = &smpte240m_coeffs;
break;
}
if (params->quantization == V4L2_QUANTIZATION_FULL_RANGE) {
csm = coeffs->full;
rkisp1_param_set_bits(params, RKISP1_CIF_ISP_CTRL, rkisp1_param_set_bits(params, RKISP1_CIF_ISP_CTRL,
RKISP1_CIF_ISP_CTRL_ISP_CSM_Y_FULL_ENA | RKISP1_CIF_ISP_CTRL_ISP_CSM_Y_FULL_ENA |
RKISP1_CIF_ISP_CTRL_ISP_CSM_C_FULL_ENA); RKISP1_CIF_ISP_CTRL_ISP_CSM_C_FULL_ENA);
} else { } else {
for (i = 0; i < ARRAY_SIZE(limited_range_coeff); i++) csm = coeffs->limited;
rkisp1_write(params->rkisp1,
RKISP1_CIF_ISP_CC_COEFF_0 + i * 4,
limited_range_coeff[i]);
rkisp1_param_clear_bits(params, RKISP1_CIF_ISP_CTRL, rkisp1_param_clear_bits(params, RKISP1_CIF_ISP_CTRL,
RKISP1_CIF_ISP_CTRL_ISP_CSM_Y_FULL_ENA | RKISP1_CIF_ISP_CTRL_ISP_CSM_Y_FULL_ENA |
RKISP1_CIF_ISP_CTRL_ISP_CSM_C_FULL_ENA); RKISP1_CIF_ISP_CTRL_ISP_CSM_C_FULL_ENA);
} }
for (i = 0; i < 9; i++)
rkisp1_write(params->rkisp1, RKISP1_CIF_ISP_CC_COEFF_0 + i * 4,
csm[i]);
} }
/* ISP De-noise Pre-Filter(DPF) function */ /* ISP De-noise Pre-Filter(DPF) function */
@ -1574,9 +1631,11 @@ static void rkisp1_params_config_parameter(struct rkisp1_params *params)
void rkisp1_params_configure(struct rkisp1_params *params, void rkisp1_params_configure(struct rkisp1_params *params,
enum rkisp1_fmt_raw_pat_type bayer_pat, enum rkisp1_fmt_raw_pat_type bayer_pat,
enum v4l2_quantization quantization) enum v4l2_quantization quantization,
enum v4l2_ycbcr_encoding ycbcr_encoding)
{ {
params->quantization = quantization; params->quantization = quantization;
params->ycbcr_encoding = ycbcr_encoding;
params->raw_type = bayer_pat; params->raw_type = bayer_pat;
rkisp1_params_config_parameter(params); rkisp1_params_config_parameter(params);
} }