]> www.infradead.org Git - users/hch/uuid.git/commitdiff
drm/i915: Implement CTM property support for VLV
authorVille Syrjälä <ville.syrjala@linux.intel.com>
Thu, 13 Apr 2023 16:49:14 +0000 (19:49 +0300)
committerVille Syrjälä <ville.syrjala@linux.intel.com>
Fri, 26 May 2023 15:08:16 +0000 (18:08 +0300)
VLV has a so called "wide gamut color correction" unit (WGC).
What it is is a 3x3 matrix similar to the later CHV CGM
CSC, with less precisions/range. In fact CHV also has the WGC
but using it there doesn't really make sense when you have the
superior CGM CSC around.

Hook up the necessary stuff to expose the WGC as the CTM
crtc property.

One additional crazy idea that came to mind would be to use
the WGC as an output CSC on CHV for YCbCr output. But it
would be incompatible with the legacy LUT usage. In fact
since the WGC lacks post-offsets we'd probably have to
use the legacy LUT to do that final part of the RGB->YCbCr
conversion. Sounds doable, but perhaps not worth the hassle.

Signed-off-by: Ville Syrjälä <ville.syrjala@linux.intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20230413164916.4221-6-ville.syrjala@linux.intel.com
Reviewed-by: Uma Shankar <uma.shankar@intel.com>
drivers/gpu/drm/i915/display/intel_color.c
drivers/gpu/drm/i915/display/intel_crtc_state_dump.c
drivers/gpu/drm/i915/display/intel_display.c
drivers/gpu/drm/i915/display/intel_display_types.h
drivers/gpu/drm/i915/i915_reg.h

index 9581dc912d8eaa70ed8f8906d2ac84670c19a156..8966e65605168fb1798fa42634726596b72154b0 100644 (file)
@@ -587,6 +587,98 @@ static u16 ctm_to_twos_complement(u64 coeff, int int_bits, int frac_bits)
        return c & (BIT(int_bits + frac_bits) - 1);
 }
 
+/*
+ * VLV/CHV Wide Gamut Color Correction (WGC) CSC
+ * |r|   | c0 c1 c2 |   |r|
+ * |g| = | c3 c4 c5 | x |g|
+ * |b|   | c6 c7 c8 |   |b|
+ *
+ * Coefficients are two's complement s2.10.
+ */
+static void vlv_wgc_csc_convert_ctm(const struct intel_crtc_state *crtc_state,
+                                   struct intel_csc_matrix *csc)
+{
+       const struct drm_color_ctm *ctm = crtc_state->hw.ctm->data;
+       int i;
+
+       for (i = 0; i < 9; i++)
+               csc->coeff[i] = ctm_to_twos_complement(ctm->matrix[i], 2, 10);
+}
+
+static void vlv_load_wgc_csc(struct intel_crtc *crtc,
+                            const struct intel_csc_matrix *csc)
+{
+       struct drm_i915_private *dev_priv = to_i915(crtc->base.dev);
+       enum pipe pipe = crtc->pipe;
+
+       intel_de_write_fw(dev_priv, PIPE_WGC_C01_C00(pipe),
+                         csc->coeff[1] << 16 | csc->coeff[0]);
+       intel_de_write_fw(dev_priv, PIPE_WGC_C02(pipe),
+                         csc->coeff[2]);
+
+       intel_de_write_fw(dev_priv, PIPE_WGC_C11_C10(pipe),
+                         csc->coeff[4] << 16 | csc->coeff[3]);
+       intel_de_write_fw(dev_priv, PIPE_WGC_C12(pipe),
+                         csc->coeff[5]);
+
+       intel_de_write_fw(dev_priv, PIPE_WGC_C21_C20(pipe),
+                         csc->coeff[7] << 16 | csc->coeff[6]);
+       intel_de_write_fw(dev_priv, PIPE_WGC_C22(pipe),
+                         csc->coeff[8]);
+}
+
+static void vlv_read_wgc_csc(struct intel_crtc *crtc,
+                            struct intel_csc_matrix *csc)
+{
+       struct drm_i915_private *dev_priv = to_i915(crtc->base.dev);
+       enum pipe pipe = crtc->pipe;
+       u32 tmp;
+
+       tmp = intel_de_read_fw(dev_priv, PIPE_WGC_C01_C00(pipe));
+       csc->coeff[0] = tmp & 0xffff;
+       csc->coeff[1] = tmp >> 16;
+
+       tmp = intel_de_read_fw(dev_priv, PIPE_WGC_C02(pipe));
+       csc->coeff[2] = tmp & 0xffff;
+
+       tmp = intel_de_read_fw(dev_priv, PIPE_WGC_C11_C10(pipe));
+       csc->coeff[3] = tmp & 0xffff;
+       csc->coeff[4] = tmp >> 16;
+
+       tmp = intel_de_read_fw(dev_priv, PIPE_WGC_C12(pipe));
+       csc->coeff[5] = tmp & 0xffff;
+
+       tmp = intel_de_read_fw(dev_priv, PIPE_WGC_C21_C20(pipe));
+       csc->coeff[6] = tmp & 0xffff;
+       csc->coeff[7] = tmp >> 16;
+
+       tmp = intel_de_read_fw(dev_priv, PIPE_WGC_C22(pipe));
+       csc->coeff[8] = tmp & 0xffff;
+}
+
+static void vlv_read_csc(struct intel_crtc_state *crtc_state)
+{
+       struct intel_crtc *crtc = to_intel_crtc(crtc_state->uapi.crtc);
+
+       if (crtc_state->wgc_enable)
+               vlv_read_wgc_csc(crtc, &crtc_state->csc);
+}
+
+static void vlv_assign_csc(struct intel_crtc_state *crtc_state)
+{
+       struct drm_i915_private *i915 = to_i915(crtc_state->uapi.crtc->dev);
+
+       if (crtc_state->hw.ctm) {
+               drm_WARN_ON(&i915->drm, !crtc_state->wgc_enable);
+
+               vlv_wgc_csc_convert_ctm(crtc_state, &crtc_state->csc);
+       } else {
+               drm_WARN_ON(&i915->drm, crtc_state->wgc_enable);
+
+               intel_csc_clear(&crtc_state->csc);
+       }
+}
+
 /*
  * CHV Color Gamut Mapping (CGM) CSC
  * |r|   | c0 c1 c2 |   |r|
@@ -672,6 +764,8 @@ static void chv_assign_csc(struct intel_crtc_state *crtc_state)
 {
        struct drm_i915_private *i915 = to_i915(crtc_state->uapi.crtc->dev);
 
+       drm_WARN_ON(&i915->drm, crtc_state->wgc_enable);
+
        if (crtc_state->hw.ctm) {
                drm_WARN_ON(&i915->drm, (crtc_state->cgm_mode & CGM_PIPE_MODE_CSC) == 0);
 
@@ -1574,6 +1668,16 @@ static void icl_load_luts(const struct intel_crtc_state *crtc_state)
        }
 }
 
+static void vlv_load_luts(const struct intel_crtc_state *crtc_state)
+{
+       struct intel_crtc *crtc = to_intel_crtc(crtc_state->uapi.crtc);
+
+       if (crtc_state->wgc_enable)
+               vlv_load_wgc_csc(crtc, &crtc_state->csc);
+
+       i965_load_luts(crtc_state);
+}
+
 static u32 chv_cgm_degamma_ldw(const struct drm_color_lut *color)
 {
        return REG_FIELD_PREP(CGM_PIPE_DEGAMMA_GREEN_LDW_MASK, drm_color_lut_extract(color->green, 14)) |
@@ -1728,6 +1832,18 @@ static bool intel_can_preload_luts(const struct intel_crtc_state *new_crtc_state
                !old_crtc_state->pre_csc_lut;
 }
 
+static bool vlv_can_preload_luts(const struct intel_crtc_state *new_crtc_state)
+{
+       struct intel_crtc *crtc = to_intel_crtc(new_crtc_state->uapi.crtc);
+       struct intel_atomic_state *state =
+               to_intel_atomic_state(new_crtc_state->uapi.state);
+       const struct intel_crtc_state *old_crtc_state =
+               intel_atomic_get_old_crtc_state(state, crtc);
+
+       return !old_crtc_state->wgc_enable &&
+               !old_crtc_state->post_csc_lut;
+}
+
 static bool chv_can_preload_luts(const struct intel_crtc_state *new_crtc_state)
 {
        struct intel_crtc *crtc = to_intel_crtc(new_crtc_state->uapi.crtc);
@@ -1744,7 +1860,7 @@ static bool chv_can_preload_luts(const struct intel_crtc_state *new_crtc_state)
        if (old_crtc_state->cgm_mode || new_crtc_state->cgm_mode)
                return false;
 
-       return !old_crtc_state->post_csc_lut;
+       return vlv_can_preload_luts(new_crtc_state);
 }
 
 int intel_color_check(struct intel_crtc_state *crtc_state)
@@ -2031,6 +2147,39 @@ static int i9xx_color_check(struct intel_crtc_state *crtc_state)
        return 0;
 }
 
+/*
+ * VLV color pipeline:
+ * u0.10 -> WGC csc -> u0.10 -> pipe gamma -> u0.10
+ */
+static int vlv_color_check(struct intel_crtc_state *crtc_state)
+{
+       int ret;
+
+       ret = check_luts(crtc_state);
+       if (ret)
+               return ret;
+
+       crtc_state->gamma_enable =
+               crtc_state->hw.gamma_lut &&
+               !crtc_state->c8_planes;
+
+       crtc_state->gamma_mode = i9xx_gamma_mode(crtc_state);
+
+       crtc_state->wgc_enable = crtc_state->hw.ctm;
+
+       ret = intel_color_add_affected_planes(crtc_state);
+       if (ret)
+               return ret;
+
+       intel_assign_luts(crtc_state);
+
+       vlv_assign_csc(crtc_state);
+
+       crtc_state->preload_luts = vlv_can_preload_luts(crtc_state);
+
+       return 0;
+}
+
 static u32 chv_cgm_mode(const struct intel_crtc_state *crtc_state)
 {
        u32 cgm_mode = 0;
@@ -2081,6 +2230,12 @@ static int chv_color_check(struct intel_crtc_state *crtc_state)
 
        crtc_state->cgm_mode = chv_cgm_mode(crtc_state);
 
+       /*
+        * We always bypass the WGC CSC and use the CGM CSC
+        * instead since it has degamma and better precision.
+        */
+       crtc_state->wgc_enable = false;
+
        ret = intel_color_add_affected_planes(crtc_state);
        if (ret)
                return ret;
@@ -3395,6 +3550,15 @@ static const struct intel_color_funcs chv_color_funcs = {
        .read_csc = chv_read_csc,
 };
 
+static const struct intel_color_funcs vlv_color_funcs = {
+       .color_check = vlv_color_check,
+       .color_commit_arm = i9xx_color_commit_arm,
+       .load_luts = vlv_load_luts,
+       .read_luts = i965_read_luts,
+       .lut_equal = i965_lut_equal,
+       .read_csc = vlv_read_csc,
+};
+
 static const struct intel_color_funcs i965_color_funcs = {
        .color_check = i9xx_color_check,
        .color_commit_arm = i9xx_color_commit_arm,
@@ -3502,7 +3666,7 @@ void intel_color_crtc_init(struct intel_crtc *crtc)
 
        gamma_lut_size = DISPLAY_INFO(i915)->color.gamma_lut_size;
        degamma_lut_size = DISPLAY_INFO(i915)->color.degamma_lut_size;
-       has_ctm = DISPLAY_VER(i915) >= 5 && !IS_VALLEYVIEW(i915);
+       has_ctm = DISPLAY_VER(i915) >= 5;
 
        /*
         * "DPALETTE_A: NOTE: The 8-bit (non-10-bit) mode is the
@@ -3541,6 +3705,8 @@ void intel_color_init_hooks(struct drm_i915_private *i915)
        if (HAS_GMCH(i915)) {
                if (IS_CHERRYVIEW(i915))
                        i915->display.funcs.color = &chv_color_funcs;
+               else if (IS_VALLEYVIEW(i915))
+                       i915->display.funcs.color = &vlv_color_funcs;
                else if (DISPLAY_VER(i915) >= 4)
                        i915->display.funcs.color = &i965_color_funcs;
                else
index 27d7bab4642742fe20ea987d56a865bd1b5ea2eb..8d4640d0fd346b51fdb599ff79759a7ceebdf49c 100644 (file)
@@ -184,7 +184,7 @@ ilk_dump_csc(struct drm_i915_private *i915, const char *name,
 }
 
 static void
-chv_dump_csc(struct drm_i915_private *i915, const char *name,
+vlv_dump_csc(struct drm_i915_private *i915, const char *name,
             const struct intel_csc_matrix *csc)
 {
        int i;
@@ -371,7 +371,9 @@ void intel_crtc_state_dump(const struct intel_crtc_state *pipe_config,
        if (!HAS_GMCH(i915))
                ilk_dump_csc(i915, "pipe csc", &pipe_config->csc);
        else if (IS_CHERRYVIEW(i915))
-               chv_dump_csc(i915, "cgm csc", &pipe_config->csc);
+               vlv_dump_csc(i915, "cgm csc", &pipe_config->csc);
+       else if (IS_VALLEYVIEW(i915))
+               vlv_dump_csc(i915, "wgc csc", &pipe_config->csc);
 
 dump_planes:
        if (!state)
index 38836e2e8807e495e0edb59cae9b7cb1334dd08f..f51a55f4e9d0a717c4c5b7d056d55c41b0fe409c 100644 (file)
@@ -2740,6 +2740,9 @@ void i9xx_set_pipeconf(const struct intel_crtc_state *crtc_state)
 
        val |= TRANSCONF_GAMMA_MODE(crtc_state->gamma_mode);
 
+       if (crtc_state->wgc_enable)
+               val |= TRANSCONF_WGC_ENABLE;
+
        val |= TRANSCONF_FRAME_START_DELAY(crtc_state->framestart_delay - 1);
 
        intel_de_write(dev_priv, TRANSCONF(cpu_transcoder), val);
@@ -2934,6 +2937,10 @@ static bool i9xx_get_pipe_config(struct intel_crtc *crtc,
 
        pipe_config->framestart_delay = REG_FIELD_GET(TRANSCONF_FRAME_START_DELAY_MASK, tmp) + 1;
 
+       if ((IS_VALLEYVIEW(dev_priv) || IS_CHERRYVIEW(dev_priv)) &&
+           (tmp & TRANSCONF_WGC_ENABLE))
+               pipe_config->wgc_enable = true;
+
        if (IS_CHERRYVIEW(dev_priv))
                pipe_config->cgm_mode = intel_de_read(dev_priv,
                                                      CGM_PIPE_MODE(crtc->pipe));
@@ -5277,6 +5284,7 @@ intel_pipe_config_compare(const struct intel_crtc_state *current_config,
                        PIPE_CONF_CHECK_X(csc_mode);
                PIPE_CONF_CHECK_BOOL(gamma_enable);
                PIPE_CONF_CHECK_BOOL(csc_enable);
+               PIPE_CONF_CHECK_BOOL(wgc_enable);
 
                PIPE_CONF_CHECK_I(linetime);
                PIPE_CONF_CHECK_I(ips_linetime);
index 88b2a55d19f21fa3736e310f1b14e00865ba388c..2d8297f8d08895e29bfe26b15af561017270b5b5 100644 (file)
@@ -1343,6 +1343,9 @@ struct intel_crtc_state {
        /* enable pipe csc? */
        bool csc_enable;
 
+       /* enable vlv/chv wgc csc? */
+       bool wgc_enable;
+
        /* big joiner pipe bitmask */
        u8 bigjoiner_pipes;
 
index f1ba1eae26ca58054741fb735ecd20960063fb23..0523418129c51db5cc464be7e2e624707a03d228 100644 (file)
 #define   TRANSCONF_MSA_TIMING_DELAY_MASK      REG_GENMASK(19, 18) /* ilk/snb/ivb */
 #define   TRANSCONF_MSA_TIMING_DELAY(x)                REG_FIELD_PREP(TRANSCONF_MSA_TIMING_DELAY_MASK, (x))
 #define   TRANSCONF_CXSR_DOWNCLOCK             REG_BIT(16)
+#define   TRANSCONF_WGC_ENABLE                 REG_BIT(15) /* vlv/chv only */
 #define   TRANSCONF_REFRESH_RATE_ALT_VLV               REG_BIT(14)
 #define   TRANSCONF_COLOR_RANGE_SELECT         REG_BIT(13)
 #define   TRANSCONF_OUTPUT_COLORSPACE_MASK     REG_GENMASK(12, 11) /* ilk-ivb */
@@ -6438,6 +6439,20 @@ enum skl_power_gate {
                                                            (index) * 4, _PLANE_CSC_POSTOFF_HI_2(pipe) + \
                                                            (index) * 4)
 
+#define _PIPE_A_WGC_C01_C00    0x600B0 /* s2.10 */
+#define _PIPE_A_WGC_C02                0x600B4 /* s2.10 */
+#define _PIPE_A_WGC_C11_C10    0x600B8 /* s2.10 */
+#define _PIPE_A_WGC_C12                0x600BC /* s2.10 */
+#define _PIPE_A_WGC_C21_C20    0x600C0 /* s2.10 */
+#define _PIPE_A_WGC_C22                0x600C4 /* s2.10 */
+
+#define PIPE_WGC_C01_C00(pipe)         _MMIO_TRANS2(pipe, _PIPE_A_WGC_C01_C00)
+#define PIPE_WGC_C02(pipe)             _MMIO_TRANS2(pipe, _PIPE_A_WGC_C02)
+#define PIPE_WGC_C11_C10(pipe)         _MMIO_TRANS2(pipe, _PIPE_A_WGC_C11_C10)
+#define PIPE_WGC_C12(pipe)             _MMIO_TRANS2(pipe, _PIPE_A_WGC_C12)
+#define PIPE_WGC_C21_C20(pipe)         _MMIO_TRANS2(pipe, _PIPE_A_WGC_C21_C20)
+#define PIPE_WGC_C22(pipe)             _MMIO_TRANS2(pipe, _PIPE_A_WGC_C22)
+
 /* pipe CSC & degamma/gamma LUTs on CHV */
 #define _CGM_PIPE_A_CSC_COEFF01        (VLV_DISPLAY_BASE + 0x67900)
 #define _CGM_PIPE_A_CSC_COEFF23        (VLV_DISPLAY_BASE + 0x67904)