media: rkisp1: Configure CSM based on YCbCr encoding
authorLaurent Pinchart <laurent.pinchart@ideasonboard.com>
Fri, 12 Aug 2022 22:44:14 +0000 (00:44 +0200)
committerMauro Carvalho Chehab <mchehab@kernel.org>
Sat, 24 Sep 2022 06:37:19 +0000 (08:37 +0200)
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>
drivers/media/platform/rockchip/rkisp1/rkisp1-common.h
drivers/media/platform/rockchip/rkisp1/rkisp1-isp.c
drivers/media/platform/rockchip/rkisp1/rkisp1-params.c

index 8056997d5c29a96282745a4d767548f7e8b71c68..b704e955cb2872e83286a85201dd3299691a8184 100644 (file)
@@ -378,6 +378,7 @@ struct rkisp1_params {
        struct v4l2_format vdev_fmt;
 
        enum v4l2_quantization quantization;
+       enum v4l2_ycbcr_encoding ycbcr_encoding;
        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.
  * @bayer_pat:   the bayer pattern on the isp video sink 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,
                             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.
  *                        This function is called by the isp entity upon stream start
index 51134f642ef970c69e2ece5a0ffc407df03ee4a7..f19c0718963fe3a8e9d3d649bbc35dc01a8de133 100644 (file)
@@ -234,7 +234,8 @@ static int rkisp1_config_isp(struct rkisp1_isp *isp,
                                                 RKISP1_ISP_PAD_SOURCE_VIDEO,
                                                 V4L2_SUBDEV_FORMAT_ACTIVE);
                rkisp1_params_configure(&rkisp1->params, sink_fmt->bayer_pat,
-                                       src_frm->quantization);
+                                       src_frm->quantization,
+                                       src_frm->ycbcr_enc);
        }
 
        return 0;
index 163419624370fff02b89b5f8b80d348f0d994401..246a6faa1fc1b288efeef82dc9d48ad0458915d8 100644 (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 const u16 full_range_coeff[] = {
-               0x0026, 0x004b, 0x000f,
-               0x01ea, 0x01d6, 0x0040,
-               0x0040, 0x01ca, 0x01f6
+       struct csm_coeffs {
+               u16 limited[9];
+               u16 full[9];
        };
-       static const u16 limited_range_coeff[] = {
-               0x0021, 0x0040, 0x000d,
-               0x01ed, 0x01db, 0x0038,
-               0x0038, 0x01d1, 0x01f7,
+       static const struct csm_coeffs rec601_coeffs = {
+               .limited = {
+                       0x0021, 0x0042, 0x000d,
+                       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;
 
-       if (params->quantization == V4L2_QUANTIZATION_FULL_RANGE) {
-               for (i = 0; i < ARRAY_SIZE(full_range_coeff); i++)
-                       rkisp1_write(params->rkisp1,
-                                    RKISP1_CIF_ISP_CC_COEFF_0 + i * 4,
-                                    full_range_coeff[i]);
+       switch (params->ycbcr_encoding) {
+       case V4L2_YCBCR_ENC_601:
+       default:
+               coeffs = &rec601_coeffs;
+               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_CIF_ISP_CTRL_ISP_CSM_Y_FULL_ENA |
                                      RKISP1_CIF_ISP_CTRL_ISP_CSM_C_FULL_ENA);
        } else {
-               for (i = 0; i < ARRAY_SIZE(limited_range_coeff); i++)
-                       rkisp1_write(params->rkisp1,
-                                    RKISP1_CIF_ISP_CC_COEFF_0 + i * 4,
-                                    limited_range_coeff[i]);
-
+               csm = coeffs->limited;
                rkisp1_param_clear_bits(params, RKISP1_CIF_ISP_CTRL,
                                        RKISP1_CIF_ISP_CTRL_ISP_CSM_Y_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 */
@@ -1574,9 +1631,11 @@ static void rkisp1_params_config_parameter(struct rkisp1_params *params)
 
 void rkisp1_params_configure(struct rkisp1_params *params,
                             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->ycbcr_encoding = ycbcr_encoding;
        params->raw_type = bayer_pat;
        rkisp1_params_config_parameter(params);
 }