drm/i915: support 3 pipes on IVB+
authorJesse Barnes <jbarnes@virtuousgeek.org>
Fri, 2 Sep 2011 19:54:37 +0000 (12:54 -0700)
committerKeith Packard <keithp@keithp.com>
Thu, 20 Oct 2011 22:26:42 +0000 (15:26 -0700)
Well almost anyway.  IVB has 3 planes, pipes, transcoders, and FDI
interfaces, but only 2 pipe PLLs.  So two of the pipes must use the same
pipe timings (e.g. 2 DP plus one other, or two HDMI with the same mode
and one other, etc.).

Signed-off-by: Jesse Barnes <jbarnes@virtuousgeek.org>
Tested-By: Eugeni Dodonov <eugeni.dodonov@intel.com>
Reviewed-By: Eugeni Dodonov <eugeni.dodonov@intel.com>
Signed-off-by: Keith Packard <keithp@keithp.com>
drivers/gpu/drm/i915/i915_dma.c
drivers/gpu/drm/i915/i915_drv.h
drivers/gpu/drm/i915/intel_display.c
drivers/gpu/drm/i915/intel_dp.c
drivers/gpu/drm/i915/intel_hdmi.c
drivers/gpu/drm/i915/intel_lvds.c
drivers/gpu/drm/i915/intel_sdvo.c

index d76da389f52174633050bd66a73d023a2869aa15..2eac955dee187337e7a4ff0e4f309a8aa8ce66a4 100644 (file)
@@ -2035,7 +2035,9 @@ int i915_driver_load(struct drm_device *dev, unsigned long flags)
        spin_lock_init(&dev_priv->error_lock);
        spin_lock_init(&dev_priv->rps_lock);
 
-       if (IS_MOBILE(dev) || !IS_GEN2(dev))
+       if (IS_IVYBRIDGE(dev))
+               dev_priv->num_pipe = 3;
+       else if (IS_MOBILE(dev) || !IS_GEN2(dev))
                dev_priv->num_pipe = 2;
        else
                dev_priv->num_pipe = 1;
index c02cdace744ca231162cf8efdc5c16c0b2fd000d..c5ca0d37f089e598a4a2fb4d4454b6b818d9f777 100644 (file)
@@ -674,8 +674,8 @@ typedef struct drm_i915_private {
        /* Panel fitter placement and size for Ironlake+ */
        u32 pch_pf_pos, pch_pf_size;
 
-       struct drm_crtc *plane_to_crtc_mapping[2];
-       struct drm_crtc *pipe_to_crtc_mapping[2];
+       struct drm_crtc *plane_to_crtc_mapping[3];
+       struct drm_crtc *pipe_to_crtc_mapping[3];
        wait_queue_head_t pending_flip_queue;
        bool flip_pending_is_done;
 
index dade95ca0d8682ed17467fe1dbbe3895b0c0531b..6b1a1edeafef45afa60350d7e94c6683ed52b06c 100644 (file)
@@ -2092,6 +2092,7 @@ static int ironlake_update_plane(struct drm_crtc *crtc,
        switch (plane) {
        case 0:
        case 1:
+       case 2:
                break;
        default:
                DRM_ERROR("Can't update plane %d in SAREA\n", plane);
@@ -2191,6 +2192,10 @@ intel_pipe_set_base(struct drm_crtc *crtc, int x, int y,
        case 0:
        case 1:
                break;
+       case 2:
+               if (IS_IVYBRIDGE(dev))
+                       break;
+               /* fall through otherwise */
        default:
                DRM_ERROR("no plane for crtc\n");
                return -EINVAL;
@@ -2889,6 +2894,8 @@ static void ironlake_pch_enable(struct drm_crtc *crtc)
                        temp |= (TRANSA_DPLL_ENABLE | TRANSA_DPLLA_SEL);
                else if (pipe == 1 && (temp & TRANSB_DPLL_ENABLE) == 0)
                        temp |= (TRANSB_DPLL_ENABLE | TRANSB_DPLLB_SEL);
+               else if (pipe == 2 && (temp & TRANSC_DPLL_ENABLE) == 0)
+                       temp |= (TRANSC_DPLL_ENABLE | TRANSC_DPLLB_SEL);
                I915_WRITE(PCH_DPLL_SEL, temp);
        }
 
index 3d73374c20d11e67a0064bb28a5bb5b3f2e4fc9a..3009d2aaaa3a79198495e44b073adf92aed293ba 100644 (file)
@@ -2215,7 +2215,7 @@ intel_dp_init(struct drm_device *dev, int output_reg)
                                  ironlake_panel_vdd_work);
        }
 
-       intel_encoder->crtc_mask = (1 << 0) | (1 << 1);
+       intel_encoder->crtc_mask = (1 << 0) | (1 << 1) | (1 << 2);
        connector->interlace_allowed = true;
        connector->doublescan_allowed = 0;
 
index 75026ba41a8eb21d345ee50d6077c301feb03ec3..185c5aa1bb25b50563615e1bb1eb0697b46bd6f4 100644 (file)
@@ -514,7 +514,7 @@ void intel_hdmi_init(struct drm_device *dev, int sdvox_reg)
        connector->polled = DRM_CONNECTOR_POLL_HPD;
        connector->interlace_allowed = 0;
        connector->doublescan_allowed = 0;
-       intel_encoder->crtc_mask = (1 << 0) | (1 << 1);
+       intel_encoder->crtc_mask = (1 << 0) | (1 << 1) | (1 << 2);
 
        /* Set up the DDC bus. */
        if (sdvox_reg == SDVOB) {
index 31da77f5c051b816f705618a950ea9dfe3d50ee9..42f165a520de440880b574c70f1ca4aa4b7b5282 100644 (file)
@@ -888,9 +888,11 @@ bool intel_lvds_init(struct drm_device *dev)
        intel_encoder->type = INTEL_OUTPUT_LVDS;
 
        intel_encoder->clone_mask = (1 << INTEL_LVDS_CLONE_BIT);
-       intel_encoder->crtc_mask = (1 << 1);
-       if (INTEL_INFO(dev)->gen >= 5)
-               intel_encoder->crtc_mask |= (1 << 0);
+       if (HAS_PCH_SPLIT(dev))
+               intel_encoder->crtc_mask = (1 << 0) | (1 << 1) | (1 << 2);
+       else
+               intel_encoder->crtc_mask = (1 << 1);
+
        drm_encoder_helper_add(encoder, &intel_lvds_helper_funcs);
        drm_connector_helper_add(connector, &intel_lvds_connector_helper_funcs);
        connector->display_info.subpixel_order = SubPixelHorizontalRGB;
index 1b97d1f48fb012dd803c61e5f177adf8bcf5ecbf..6db3b1ccb6ebda273255f50bfd931f5e7acd14de 100644 (file)
@@ -2203,7 +2203,7 @@ intel_sdvo_output_setup(struct intel_sdvo *intel_sdvo, uint16_t flags)
                              bytes[0], bytes[1]);
                return false;
        }
-       intel_sdvo->base.crtc_mask = (1 << 0) | (1 << 1);
+       intel_sdvo->base.crtc_mask = (1 << 0) | (1 << 1) | (1 << 2);
 
        return true;
 }