|
@@ -41,6 +41,22 @@
|
|
|
|
|
|
#define LEGACY_LUT_LENGTH (sizeof(struct drm_color_lut) * 256)
|
|
|
|
|
|
+/* Post offset values for RGB->YCBCR conversion */
|
|
|
+#define POSTOFF_RGB_TO_YUV_HI 0x800
|
|
|
+#define POSTOFF_RGB_TO_YUV_ME 0x100
|
|
|
+#define POSTOFF_RGB_TO_YUV_LO 0x800
|
|
|
+
|
|
|
+/*
|
|
|
+ * These values are direct register values specified in the Bspec,
|
|
|
+ * for RGB->YUV conversion matrix (colorspace BT709)
|
|
|
+ */
|
|
|
+#define CSC_RGB_TO_YUV_RU_GU 0x2ba809d8
|
|
|
+#define CSC_RGB_TO_YUV_BU 0x37e80000
|
|
|
+#define CSC_RGB_TO_YUV_RY_GY 0x1e089cc0
|
|
|
+#define CSC_RGB_TO_YUV_BY 0xb5280000
|
|
|
+#define CSC_RGB_TO_YUV_RV_GV 0xbce89ad8
|
|
|
+#define CSC_RGB_TO_YUV_BV 0x1e080000
|
|
|
+
|
|
|
/*
|
|
|
* Extract the CSC coefficient from a CTM coefficient (in U32.32 fixed point
|
|
|
* format). This macro takes the coefficient we want transformed and the
|
|
@@ -91,6 +107,30 @@ static void ctm_mult_by_limited(uint64_t *result, int64_t *input)
|
|
|
}
|
|
|
}
|
|
|
|
|
|
+void i9xx_load_ycbcr_conversion_matrix(struct intel_crtc *intel_crtc)
|
|
|
+{
|
|
|
+ int pipe = intel_crtc->pipe;
|
|
|
+ struct drm_i915_private *dev_priv = to_i915(intel_crtc->base.dev);
|
|
|
+
|
|
|
+ I915_WRITE(PIPE_CSC_PREOFF_HI(pipe), 0);
|
|
|
+ I915_WRITE(PIPE_CSC_PREOFF_ME(pipe), 0);
|
|
|
+ I915_WRITE(PIPE_CSC_PREOFF_LO(pipe), 0);
|
|
|
+
|
|
|
+ I915_WRITE(PIPE_CSC_COEFF_RU_GU(pipe), CSC_RGB_TO_YUV_RU_GU);
|
|
|
+ I915_WRITE(PIPE_CSC_COEFF_BU(pipe), CSC_RGB_TO_YUV_BU);
|
|
|
+
|
|
|
+ I915_WRITE(PIPE_CSC_COEFF_RY_GY(pipe), CSC_RGB_TO_YUV_RY_GY);
|
|
|
+ I915_WRITE(PIPE_CSC_COEFF_BY(pipe), CSC_RGB_TO_YUV_BY);
|
|
|
+
|
|
|
+ I915_WRITE(PIPE_CSC_COEFF_RV_GV(pipe), CSC_RGB_TO_YUV_RV_GV);
|
|
|
+ I915_WRITE(PIPE_CSC_COEFF_BV(pipe), CSC_RGB_TO_YUV_BV);
|
|
|
+
|
|
|
+ I915_WRITE(PIPE_CSC_POSTOFF_HI(pipe), POSTOFF_RGB_TO_YUV_HI);
|
|
|
+ I915_WRITE(PIPE_CSC_POSTOFF_ME(pipe), POSTOFF_RGB_TO_YUV_ME);
|
|
|
+ I915_WRITE(PIPE_CSC_POSTOFF_LO(pipe), POSTOFF_RGB_TO_YUV_LO);
|
|
|
+ I915_WRITE(PIPE_CSC_MODE(pipe), 0);
|
|
|
+}
|
|
|
+
|
|
|
/* Set up the pipe CSC unit. */
|
|
|
static void i9xx_load_csc_matrix(struct drm_crtc_state *crtc_state)
|
|
|
{
|
|
@@ -101,7 +141,10 @@ static void i9xx_load_csc_matrix(struct drm_crtc_state *crtc_state)
|
|
|
uint16_t coeffs[9] = { 0, };
|
|
|
struct intel_crtc_state *intel_crtc_state = to_intel_crtc_state(crtc_state);
|
|
|
|
|
|
- if (crtc_state->ctm) {
|
|
|
+ if (intel_crtc_state->ycbcr420) {
|
|
|
+ i9xx_load_ycbcr_conversion_matrix(intel_crtc);
|
|
|
+ return;
|
|
|
+ } else if (crtc_state->ctm) {
|
|
|
struct drm_color_ctm *ctm =
|
|
|
(struct drm_color_ctm *)crtc_state->ctm->data;
|
|
|
uint64_t input[9] = { 0, };
|