drm/amd/display: Double buffer dcn2 Gamut Remap
authorNoah Abradjian <noah.abradjian@amd.com>
Mon, 16 Dec 2019 15:50:53 +0000 (10:50 -0500)
committerAlex Deucher <alexander.deucher@amd.com>
Thu, 16 Jan 2020 19:13:30 +0000 (14:13 -0500)
[Why]
When rapidly adjusting color temperature, screen tearing was observed.
This was due to overwritten values in gamut remap registers.
This issue was solved for OCSC and ICSC by alternating between "A" and
"B" registers to double buffer the writes.

[How]
Create new set_gamut_remap and program_gamut_remap for dcn20.
Alternate which registers are written to by switching modes each time.
Also fixes ICSC mode reg read to use proper data offset.

Signed-off-by: Noah Abradjian <noah.abradjian@amd.com>
Reviewed-by: Tony Cheng <Tony.Cheng@amd.com>
Acked-by: Harry Wentland <harry.wentland@amd.com>
Acked-by: Rodrigo Siqueira <Rodrigo.Siqueira@amd.com>
Signed-off-by: Alex Deucher <alexander.deucher@amd.com>
drivers/gpu/drm/amd/display/dc/dcn20/dcn20_dpp.c
drivers/gpu/drm/amd/display/dc/dcn20/dcn20_dpp.h
drivers/gpu/drm/amd/display/dc/dcn20/dcn20_dpp_cm.c

index bbdab5000a7ccc178e1f8b3cd2be05e4bcabe216..13e057d7ee93da28efd01a55cbcea1873801e09b 100644 (file)
@@ -458,7 +458,7 @@ static struct dpp_funcs dcn20_dpp_funcs = {
        .dpp_reset = dpp_reset,
        .dpp_set_scaler = dpp1_dscl_set_scaler_manual_scale,
        .dpp_get_optimal_number_of_taps = dpp1_get_optimal_number_of_taps,
-       .dpp_set_gamut_remap = dpp1_cm_set_gamut_remap,
+       .dpp_set_gamut_remap = dpp2_cm_set_gamut_remap,
        .dpp_set_csc_adjustment = NULL,
        .dpp_set_csc_default = NULL,
        .dpp_program_regamma_pwl = oppn20_dummy_program_regamma_pwl,
index d9ce60b6aa6efef3e8f1bcfff1eaae9afb76cb18..141dea963db99c42918931299fc3baf6683c26fe 100644 (file)
        SRI(CM_SHAPER_LUT_INDEX, CM, id)
 
 #define TF_REG_LIST_DCN20_COMMON_APPEND(id) \
+       SRI(CM_GAMUT_REMAP_B_C11_C12, CM, id),\
+       SRI(CM_GAMUT_REMAP_B_C13_C14, CM, id),\
+       SRI(CM_GAMUT_REMAP_B_C21_C22, CM, id),\
+       SRI(CM_GAMUT_REMAP_B_C23_C24, CM, id),\
+       SRI(CM_GAMUT_REMAP_B_C31_C32, CM, id),\
+       SRI(CM_GAMUT_REMAP_B_C33_C34, CM, id),\
        SRI(CM_ICSC_B_C11_C12, CM, id), \
        SRI(CM_ICSC_B_C33_C34, CM, id)
 
 /* DPP CM debug status register:
  *
  *             Status index including current ICSC, Gamut Remap Mode is 9
- *                     ICSC Mode: [5..4]
+ *                     ICSC Mode: [4..3]
+ *                     Gamut Remap Mode: [10..9]
  */
 #define CM_TEST_DEBUG_DATA_STATUS_IDX 9
-#define CM_TEST_DEBUG_DATA_ICSC_MODE_SH 4
+#define CM_TEST_DEBUG_DATA_ICSC_MODE_SH 3
 #define CM_TEST_DEBUG_DATA_ICSC_MODE_MASK 0x3
+#define CM_TEST_DEBUG_DATA_GAMUT_REMAP_MODE_SH 9
+#define CM_TEST_DEBUG_DATA_GAMUT_REMAP_MODE_MASK 0x3
 
 #define TF_REG_FIELD_LIST_DCN2_0(type) \
        TF_REG_FIELD_LIST(type) \
@@ -646,6 +655,12 @@ struct dcn2_dpp_mask {
        uint32_t DSCL_MEM_PWR_CTRL
 
 #define DPP_DCN2_REG_VARIABLE_LIST_CM_APPEND \
+       uint32_t CM_GAMUT_REMAP_B_C11_C12; \
+       uint32_t CM_GAMUT_REMAP_B_C13_C14; \
+       uint32_t CM_GAMUT_REMAP_B_C21_C22; \
+       uint32_t CM_GAMUT_REMAP_B_C23_C24; \
+       uint32_t CM_GAMUT_REMAP_B_C31_C32; \
+       uint32_t CM_GAMUT_REMAP_B_C33_C34; \
        uint32_t CM_ICSC_B_C11_C12; \
        uint32_t CM_ICSC_B_C33_C34
 
@@ -679,6 +694,12 @@ enum dcn20_input_csc_select {
        DCN2_ICSC_SELECT_ICSC_B = 2
 };
 
+enum dcn20_gamut_remap_select {
+       DCN2_GAMUT_REMAP_BYPASS = 0,
+       DCN2_GAMUT_REMAP_COEF_A = 1,
+       DCN2_GAMUT_REMAP_COEF_B = 2
+};
+
 void dpp20_read_state(struct dpp *dpp_base,
                struct dcn_dpp_state *s);
 
@@ -690,6 +711,10 @@ void dpp2_set_degamma(
                struct dpp *dpp_base,
                enum ipp_degamma_mode mode);
 
+void dpp2_cm_set_gamut_remap(
+       struct dpp *dpp_base,
+       const struct dpp_grph_csc_adjustment *adjust);
+
 void dpp2_program_input_csc(
                struct dpp *dpp_base,
                enum dc_color_space color_space,
index 423f3daa177f25bbd76d58cccb1658f9da58f199..4047d406a74c69b6720838f4e665c542787dfe1d 100644 (file)
@@ -158,6 +158,85 @@ void dpp2_set_degamma(
        }
 }
 
+static void program_gamut_remap(
+               struct dcn20_dpp *dpp,
+               const uint16_t *regval,
+               enum dcn20_gamut_remap_select select)
+{
+       uint32_t cur_select = 0;
+       struct color_matrices_reg gam_regs;
+
+       if (regval == NULL || select == DCN2_GAMUT_REMAP_BYPASS) {
+               REG_SET(CM_GAMUT_REMAP_CONTROL, 0,
+                               CM_GAMUT_REMAP_MODE, 0);
+               return;
+       }
+
+       /* determine which gamut_remap coefficients (A or B) we are using
+        * currently. select the alternate set to double buffer
+        * the update so gamut_remap is updated on frame boundary
+        */
+       cur_select = IX_REG_READ(CM_TEST_DEBUG_INDEX, CM_TEST_DEBUG_DATA,
+                                       CM_TEST_DEBUG_DATA_STATUS_IDX);
+
+       /* IX_REG_READ reads whole reg, so isolate part we want [10..9] */
+       cur_select = (cur_select >> CM_TEST_DEBUG_DATA_GAMUT_REMAP_MODE_SH)
+                                       & CM_TEST_DEBUG_DATA_GAMUT_REMAP_MODE_MASK;
+
+       /* value stored in dbg reg will be 1 greater than mode we want */
+       if (cur_select != DCN2_GAMUT_REMAP_COEF_A)
+               select = DCN2_GAMUT_REMAP_COEF_A;
+       else
+               select = DCN2_GAMUT_REMAP_COEF_B;
+
+       gam_regs.shifts.csc_c11 = dpp->tf_shift->CM_GAMUT_REMAP_C11;
+       gam_regs.masks.csc_c11  = dpp->tf_mask->CM_GAMUT_REMAP_C11;
+       gam_regs.shifts.csc_c12 = dpp->tf_shift->CM_GAMUT_REMAP_C12;
+       gam_regs.masks.csc_c12 = dpp->tf_mask->CM_GAMUT_REMAP_C12;
+
+       if (select == DCN2_GAMUT_REMAP_COEF_A) {
+               gam_regs.csc_c11_c12 = REG(CM_GAMUT_REMAP_C11_C12);
+               gam_regs.csc_c33_c34 = REG(CM_GAMUT_REMAP_C33_C34);
+       } else {
+               gam_regs.csc_c11_c12 = REG(CM_GAMUT_REMAP_B_C11_C12);
+               gam_regs.csc_c33_c34 = REG(CM_GAMUT_REMAP_B_C33_C34);
+       }
+
+       cm_helper_program_color_matrices(
+                               dpp->base.ctx,
+                               regval,
+                               &gam_regs);
+
+       REG_SET(
+                       CM_GAMUT_REMAP_CONTROL, 0,
+                       CM_GAMUT_REMAP_MODE, select);
+
+}
+
+void dpp2_cm_set_gamut_remap(
+       struct dpp *dpp_base,
+       const struct dpp_grph_csc_adjustment *adjust)
+{
+       struct dcn20_dpp *dpp = TO_DCN20_DPP(dpp_base);
+       int i = 0;
+
+       if (adjust->gamut_adjust_type != GRAPHICS_GAMUT_ADJUST_TYPE_SW)
+               /* Bypass if type is bypass or hw */
+               program_gamut_remap(dpp, NULL, DCN2_GAMUT_REMAP_BYPASS);
+       else {
+               struct fixed31_32 arr_matrix[12];
+               uint16_t arr_reg_val[12];
+
+               for (i = 0; i < 12; i++)
+                       arr_matrix[i] = adjust->temperature_matrix[i];
+
+               convert_float_matrix(
+                       arr_reg_val, arr_matrix, 12);
+
+               program_gamut_remap(dpp, arr_reg_val, DCN2_GAMUT_REMAP_COEF_A);
+       }
+}
+
 void dpp2_program_input_csc(
                struct dpp *dpp_base,
                enum dc_color_space color_space,
@@ -199,12 +278,11 @@ void dpp2_program_input_csc(
        cur_select = IX_REG_READ(CM_TEST_DEBUG_INDEX, CM_TEST_DEBUG_DATA,
                                        CM_TEST_DEBUG_DATA_STATUS_IDX);
 
-       /* IX_REG_READ reads whole reg, so isolate part we want [5..4] */
+       /* IX_REG_READ reads whole reg, so isolate part we want [4..3] */
        cur_select = (cur_select >> CM_TEST_DEBUG_DATA_ICSC_MODE_SH)
                                        & CM_TEST_DEBUG_DATA_ICSC_MODE_MASK;
 
-       /* value stored in dbg reg will be 1 greater than mode we want */
-       if (cur_select - 1 != DCN2_ICSC_SELECT_ICSC_A)
+       if (cur_select != DCN2_ICSC_SELECT_ICSC_A)
                select = DCN2_ICSC_SELECT_ICSC_A;
        else
                select = DCN2_ICSC_SELECT_ICSC_B;