From: Paulo Zanoni <paulo.r.zanoni at intel.com> This patch implements "Display Sequences for Package C8", from the "Display Mode Set Sequence" section from the Haswell documentation. Signed-off-by: Paulo Zanoni <paulo.r.zanoni at intel.com> --- Hi This patch was tested and the machines do enter C8+, so it shows our driver is not preventing C8+. There may be additional patches in the future to allow C8+ in more cases, but they shouldn't stop this patch from being merged. Please also notice that we need i915.disable_power_well=1 because we can't allow C8+ if the power well is enabled. So even if we merge this patch, the feature won't be enabled by default. This is not really a problem and it allows us to properly test everything without needing to rebase a 600-line patch every time. I wrote this patch on top of drm-intel-next-queued + the 4 patches I already sent to this list: - 2 patches that fix CPT FDI RX polarity bugs (already reviewed by Imre) - 2 patches that enable FIFO underruns and Poison messages (not reviewed yet) Thanks, Paulo drivers/gpu/drm/i915/i915_dma.c | 1 + drivers/gpu/drm/i915/i915_drv.h | 29 +++ drivers/gpu/drm/i915/i915_reg.h | 27 ++ drivers/gpu/drm/i915/intel_crt.c | 26 +- drivers/gpu/drm/i915/intel_display.c | 459 ++++++++++++++++++++++++++++++++++ drivers/gpu/drm/i915/intel_dp.c | 11 +- drivers/gpu/drm/i915/intel_drv.h | 2 + drivers/gpu/drm/i915/intel_hdmi.c | 3 + 8 files changed, 549 insertions(+), 9 deletions(-) diff --git a/drivers/gpu/drm/i915/i915_dma.c b/drivers/gpu/drm/i915/i915_dma.c index 3b315ba..bc81ab9 100644 --- a/drivers/gpu/drm/i915/i915_dma.c +++ b/drivers/gpu/drm/i915/i915_dma.c @@ -1630,6 +1630,7 @@ int i915_driver_load(struct drm_device *dev, unsigned long flags) spin_lock_init(&dev_priv->gpu_error.lock); spin_lock_init(&dev_priv->rps.lock); mutex_init(&dev_priv->dpio_lock); + mutex_init(&dev_priv->c8_lock); mutex_init(&dev_priv->rps.hw_lock); mutex_init(&dev_priv->modeset_restore_lock); diff --git a/drivers/gpu/drm/i915/i915_drv.h b/drivers/gpu/drm/i915/i915_drv.h index b0da4ae..b64d0e3 100644 --- a/drivers/gpu/drm/i915/i915_drv.h +++ b/drivers/gpu/drm/i915/i915_drv.h @@ -504,6 +504,29 @@ struct intel_gmbus { struct drm_i915_private *dev_priv; }; +struct i915_c8_saved_registers { + u32 de_imr; + u32 de_ier; + u32 aud_imr; + u32 aud_ier; + u32 gt_imr; + u32 gt_ier; + u32 pm_imr; + u32 pm_ier; + u32 srd_imr; + u32 hotplug_ctl; + u32 err_int; + + u32 sde_imr; + u32 sde_ier; + u32 fdirx_imr; + u32 gtcpch_imr; + u32 shotplug_ctl; + u32 serr_int; + + u32 lcpll_freq; +}; + struct i915_suspend_saved_registers { u8 saveLBB; u32 saveDSPACNTR; @@ -1068,6 +1091,12 @@ typedef struct drm_i915_private { struct i915_suspend_saved_registers regfile; + struct i915_c8_saved_registers c8_regfile; + bool allowing_package_c8; + /* Wake ups should happen when allowing_package_c8 is true. */ + int c8_wakeup_refcnt; + struct mutex c8_lock; + /* Old dri1 support infrastructure, beware the dragons ya fools entering * here! */ struct i915_dri1_state dri1; diff --git a/drivers/gpu/drm/i915/i915_reg.h b/drivers/gpu/drm/i915/i915_reg.h index d954612..6808ae5 100644 --- a/drivers/gpu/drm/i915/i915_reg.h +++ b/drivers/gpu/drm/i915/i915_reg.h @@ -2013,6 +2013,8 @@ #define BLC_PWM_CPU_CTL2 0x48250 #define BLC_PWM_CPU_CTL 0x48254 +#define HSW_BLC_PWM2_CTL 0x48350 + /* PCH CTL1 is totally different, all but the below bits are reserved. CTL2 is * like the normal CTL from gen4 and earlier. Hooray for confusing naming. */ #define BLC_PWM_PCH_CTL1 0xc8250 @@ -2021,6 +2023,12 @@ #define BLM_PCH_POLARITY (1 << 29) #define BLC_PWM_PCH_CTL2 0xc8254 +#define UTIL_PIN_CTL 0x48400 +#define UTIL_PIN_ENABLE (1 << 31) + +#define PCH_GTC_CTL 0xe7000 +#define PCH_GTC_ENABLE (1 << 31) + /* TV port control */ #define TV_CTL 0x68000 /** Enables the TV encoder */ @@ -3535,6 +3543,16 @@ #define DEIIR 0x44008 #define DEIER 0x4400c +#define AUDIMR 0x44084 +#define AUDIIR 0x44088 +#define AUDIER 0x4408c + +#define SRDIMR 0x64834 +#define SRDIIR 0x64838 + +#define PCH_GTCIMR 0xe7054 +#define PCH_GTCIIR 0xe7058 + /* GT interrupt. * Note that for gen6+ the ring-specific interrupt bits do alias with the * corresponding bits in the per-ring interrupt control registers. */ @@ -4709,6 +4727,8 @@ #define SBI_SSCAUXDIV_FINALDIV2SEL(x) ((x)<<4) #define SBI_DBUFF0 0x2a00 #define SBI_DBUFF0_ENABLE (1<<0) +#define SBI_GEN0 0x1f00 +#define SBI_GEN0_ENABLE (1<<0) /* LPT PIXCLK_GATE */ #define PIXCLK_GATE 0xC6020 @@ -4774,7 +4794,14 @@ #define LCPLL_CLK_FREQ_450 (0<<26) #define LCPLL_CD_CLOCK_DISABLE (1<<25) #define LCPLL_CD2X_CLOCK_DISABLE (1<<23) +#define LCPLL_POWER_DOWN_ALLOW (1<<22) #define LCPLL_CD_SOURCE_FCLK (1<<21) +#define LCPLL_CD_SOURCE_FCLK_DONE (1<<19) + +#define D_COMP (MCHBAR_MIRROR_BASE_SNB + 0x5F0C) +#define D_COMP_RCOMP_IN_PROGRESS (1<<9) +#define D_COMP_COMP_FORCE (1<<8) +#define D_COMP_COMP_DISABLE (1<<0) /* Pipe WM_LINETIME - watermark line time */ #define PIPE_WM_LINETIME_A 0x45270 diff --git a/drivers/gpu/drm/i915/intel_crt.c b/drivers/gpu/drm/i915/intel_crt.c index 1ae2d7f..ee98731 100644 --- a/drivers/gpu/drm/i915/intel_crt.c +++ b/drivers/gpu/drm/i915/intel_crt.c @@ -588,10 +588,13 @@ static enum drm_connector_status intel_crt_detect(struct drm_connector *connector, bool force) { struct drm_device *dev = connector->dev; + struct drm_i915_private *dev_priv = dev->dev_private; struct intel_crt *crt = intel_attached_crt(connector); enum drm_connector_status status; struct intel_load_detect_pipe tmp; + hsw_package_c8_wakeup(dev_priv); + if (I915_HAS_HOTPLUG(dev)) { /* We can not rely on the HPD pin always being correctly wired * up, for example many KVM do not pass it through, and so @@ -599,23 +602,30 @@ intel_crt_detect(struct drm_connector *connector, bool force) */ if (intel_crt_detect_hotplug(connector)) { DRM_DEBUG_KMS("CRT detected via hotplug\n"); - return connector_status_connected; + status = connector_status_connected; + goto out; } else DRM_DEBUG_KMS("CRT not detected via hotplug\n"); } - if (intel_crt_detect_ddc(connector)) - return connector_status_connected; + if (intel_crt_detect_ddc(connector)) { + status = connector_status_connected; + goto out; + } /* Load detection is broken on HPD capable machines. Whoever wants a * broken monitor (without edid) to work behind a broken kvm (that fails * to have the right resistors for HP detection) needs to fix this up. * For now just bail out. */ - if (I915_HAS_HOTPLUG(dev)) - return connector_status_disconnected; + if (I915_HAS_HOTPLUG(dev)) { + status = connector_status_disconnected; + goto out; + } - if (!force) - return connector->status; + if (!force) { + status = connector->status; + goto out; + } /* for pre-945g platforms use load detect */ if (intel_get_load_detect_pipe(connector, NULL, &tmp)) { @@ -627,6 +637,8 @@ intel_crt_detect(struct drm_connector *connector, bool force) } else status = connector_status_unknown; +out: + hsw_package_c8_sleep(dev_priv); return status; } diff --git a/drivers/gpu/drm/i915/intel_display.c b/drivers/gpu/drm/i915/intel_display.c index cd9fa46..9e6d6e4 100644 --- a/drivers/gpu/drm/i915/intel_display.c +++ b/drivers/gpu/drm/i915/intel_display.c @@ -5091,6 +5091,66 @@ static void lpt_init_pch_refclk(struct drm_device *dev) mutex_unlock(&dev_priv->dpio_lock); } +/* Sequence to enable CLKOUT_DP */ +static void lpt_enable_clkout_dp(struct drm_i915_private *dev_priv) +{ + uint32_t val; + + mutex_lock(&dev_priv->dpio_lock); + + val = intel_sbi_read(dev_priv, SBI_SSCCTL, SBI_ICLK); + val &= ~SBI_SSCCTL_DISABLE; + val |= SBI_SSCCTL_PATHALT; + intel_sbi_write(dev_priv, SBI_SSCCTL, val, SBI_ICLK); + + udelay(24); + + val = intel_sbi_read(dev_priv, SBI_SSCCTL, SBI_ICLK); + val &= ~SBI_SSCCTL_PATHALT; + intel_sbi_write(dev_priv, SBI_SSCCTL, val, SBI_ICLK); + + if (IS_ULT(dev_priv->dev)) { + val = intel_sbi_read(dev_priv, SBI_GEN0, SBI_ICLK); + val |= SBI_GEN0_ENABLE; + intel_sbi_write(dev_priv, SBI_GEN0, val, SBI_ICLK); + } else { + val = intel_sbi_read(dev_priv, SBI_DBUFF0, SBI_ICLK); + val |= SBI_DBUFF0_ENABLE; + intel_sbi_write(dev_priv, SBI_DBUFF0, val, SBI_ICLK); + } + + mutex_unlock(&dev_priv->dpio_lock); +} + +/* Sequence to disable CLKOUT_DP */ +static void lpt_disable_clkout_dp(struct drm_i915_private *dev_priv) +{ + uint32_t val; + + mutex_lock(&dev_priv->dpio_lock); + + if (IS_ULT(dev_priv->dev)) { + val = intel_sbi_read(dev_priv, SBI_GEN0, SBI_ICLK); + val &= ~SBI_GEN0_ENABLE; + intel_sbi_write(dev_priv, SBI_GEN0, val, SBI_ICLK); + } else { + val = intel_sbi_read(dev_priv, SBI_DBUFF0, SBI_ICLK); + val &= ~SBI_DBUFF0_ENABLE; + intel_sbi_write(dev_priv, SBI_DBUFF0, val, SBI_ICLK); + } + + val = intel_sbi_read(dev_priv, SBI_SSCCTL, SBI_ICLK); + if (!(val & SBI_SSCCTL_PATHALT)) { + val |= SBI_SSCCTL_PATHALT; + intel_sbi_write(dev_priv, SBI_SSCCTL, val, SBI_ICLK); + udelay(32); + } + val |= SBI_SSCCTL_DISABLE; + intel_sbi_write(dev_priv, SBI_SSCCTL, val, SBI_ICLK); + + mutex_unlock(&dev_priv->dpio_lock); +} + /* * Initialize reference clocks when the driver loads */ @@ -5751,6 +5811,403 @@ static bool ironlake_get_pipe_config(struct intel_crtc *crtc, return true; } +static void hsw_disable_lcpll(struct drm_i915_private *dev_priv) +{ + uint32_t val; + + val = I915_READ(LCPLL_CTL); + + dev_priv->c8_regfile.lcpll_freq = val & LCPLL_CLK_FREQ_MASK; + + val |= LCPLL_CD_SOURCE_FCLK; + I915_WRITE(LCPLL_CTL, val); + POSTING_READ(LCPLL_CTL); + + udelay(1); + + val = I915_READ(LCPLL_CTL); + if (!(val & LCPLL_CD_SOURCE_FCLK_DONE)) + DRM_ERROR("Switching to FCLK failed\n"); + + val |= LCPLL_PLL_DISABLE; + I915_WRITE(LCPLL_CTL, val); + POSTING_READ(LCPLL_CTL); + + if (wait_for((I915_READ(LCPLL_CTL) & LCPLL_PLL_LOCK) == 0, 1)) + DRM_ERROR("LCPLL still locked\n"); + + val = I915_READ(D_COMP); + val |= D_COMP_COMP_DISABLE; + I915_WRITE(D_COMP, val); + POSTING_READ(D_COMP); + + udelay(2); + + val = I915_READ(D_COMP); + if (val & D_COMP_RCOMP_IN_PROGRESS) + DRM_ERROR("D_COMP RCOMP still in progress\n"); + + val = I915_READ(LCPLL_CTL); + val |= LCPLL_POWER_DOWN_ALLOW; + I915_WRITE(LCPLL_CTL, val); + POSTING_READ(LCPLL_CTL); +} + +static void hsw_restore_lcpll(struct drm_i915_private *dev_priv) +{ + uint32_t val; + + val = I915_READ(LCPLL_CTL); + + if ((val & LCPLL_CLK_FREQ_MASK) != dev_priv->c8_regfile.lcpll_freq) + DRM_ERROR("LCPLL frequency changed\n"); + + if (val & LCPLL_POWER_DOWN_ALLOW) { + val &= ~LCPLL_POWER_DOWN_ALLOW; + I915_WRITE(LCPLL_CTL, val); + } + + if (val & LCPLL_CD_SOURCE_FCLK) { + val = I915_READ(D_COMP); + val |= D_COMP_COMP_FORCE; + val &= ~D_COMP_COMP_DISABLE; + I915_WRITE(D_COMP, val); + + val = I915_READ(LCPLL_CTL); + val &= ~LCPLL_PLL_DISABLE; + I915_WRITE(LCPLL_CTL, val); + POSTING_READ(LCPLL_CTL); + + if (wait_for(I915_READ(LCPLL_CTL) & LCPLL_PLL_LOCK, 5)) + DRM_ERROR("LCPLL not locked yet\n"); + + val = I915_READ(LCPLL_CTL); + val &= ~LCPLL_CD_SOURCE_FCLK; + I915_WRITE(LCPLL_CTL, val); + POSTING_READ(LCPLL_CTL); + + udelay(1); + + val = I915_READ(LCPLL_CTL); + if (val & LCPLL_CD_SOURCE_FCLK_DONE) + DRM_ERROR("Switching back to LCPLL failed\n"); + } +} + +static void hsw_disable_interrupts(struct drm_i915_private *dev_priv) +{ + struct i915_c8_saved_registers *c8_regfile = &dev_priv->c8_regfile; + unsigned long irqflags; + uint32_t val, deier, sdeier; + + spin_lock_irqsave(&dev_priv->irq_lock, irqflags); + + /* TODO: clear all pending graphics interrupts. */ + val = I915_READ(DEIIR); + if (val) + DRM_ERROR("Pending interrupt: DEIIR 0x%08x\n", val); + + val = I915_READ(AUDIIR); + if (val) + DRM_ERROR("Pending interrupt: AUDIIR 0x%08x\n", val); + + val = I915_READ(GTIIR); + if (val) + DRM_ERROR("Pending interrupt: GTIIR 0x%08x\n", val); + + val = I915_READ(GEN6_PMIIR); + if (val) + DRM_ERROR("Pending interrupt: PMIIR 0x%08x\n", val); + + val = I915_READ(SRDIIR); + if (val) + DRM_ERROR("Pending interrupt: SRDIIR 0x%08x\n", val); + + val = I915_READ(SDEIIR); + if (val) + DRM_ERROR("Pending interrupt: SDEIIR 0x%08x\n", val); + + val = I915_READ(_FDI_RXA_IIR); + if (val) + DRM_ERROR("Pending interrupt: FDIRXAIIR 0x%08x\n", val); + + val = I915_READ(PCH_GTCIIR); + if (val) + DRM_ERROR("Pending interrupt: GTCPCHIIR 0x%08x\n", val); + + c8_regfile->de_imr = I915_READ(DEIMR); + c8_regfile->de_ier = I915_READ(DEIER); + c8_regfile->aud_imr = I915_READ(AUDIMR); + c8_regfile->aud_ier = I915_READ(AUDIER); + c8_regfile->gt_imr = I915_READ(GTIMR); + c8_regfile->gt_ier = I915_READ(GTIER); + c8_regfile->pm_imr = I915_READ(GEN6_PMIMR); + c8_regfile->pm_ier = I915_READ(GEN6_PMIER); + c8_regfile->srd_imr = I915_READ(SRDIMR); + c8_regfile->hotplug_ctl = I915_READ(DIGITAL_PORT_HOTPLUG_CNTRL); + c8_regfile->err_int = I915_READ(GEN7_ERR_INT); + + c8_regfile->sde_imr = I915_READ(SDEIMR); + c8_regfile->sde_ier = I915_READ(SDEIER); + c8_regfile->fdirx_imr = I915_READ(_FDI_RXA_IMR); + c8_regfile->gtcpch_imr = I915_READ(PCH_GTCIMR); + c8_regfile->shotplug_ctl = I915_READ(PCH_PORT_HOTPLUG); + c8_regfile->serr_int = I915_READ(SERR_INT); + + deier = DE_MASTER_IRQ_CONTROL | DE_PCH_EVENT_IVB; + I915_WRITE(DEIMR, ~deier); + I915_WRITE(DEIER, deier); + + I915_WRITE(AUDIMR, 0xFFFFFFFF); + I915_WRITE(AUDIER, 0); + + I915_WRITE(GTIMR, 0xFFFFFFFF); + I915_WRITE(GTIER, 0); + + I915_WRITE(GEN6_PMIMR, 0xFFFFFFFF); + I915_WRITE(GEN6_PMIER, 0); + + I915_WRITE(SRDIMR, 0xFFFFFFFF); + + I915_WRITE(DIGITAL_PORT_HOTPLUG_CNTRL, 0); + + I915_WRITE(GEN7_ERR_INT, 0xFFFFFFFF); + + sdeier = SDE_PORTD_HOTPLUG_CPT | SDE_PORTC_HOTPLUG_CPT | + SDE_PORTB_HOTPLUG_CPT | SDE_CRT_HOTPLUG_CPT; + I915_WRITE(SDEIMR, ~sdeier); + I915_WRITE(SDEIER, sdeier); + + I915_WRITE(_FDI_RXA_IMR, 0xFFFFFFFF); + I915_WRITE(PCH_GTCIMR, 0xFFFFFFFF); + + I915_WRITE(PCH_PORT_HOTPLUG, PORTD_HOTPLUG_ENABLE | + PORTC_HOTPLUG_ENABLE | + PORTB_HOTPLUG_ENABLE); + I915_WRITE(SERR_INT, 0xFFFFFFFF); + + spin_unlock_irqrestore(&dev_priv->irq_lock, irqflags); +} + +static void hsw_restore_interrupts(struct drm_i915_private *dev_priv) +{ + struct i915_c8_saved_registers *c8_regfile = &dev_priv->c8_regfile; + unsigned long irqflags; + + spin_lock_irqsave(&dev_priv->irq_lock, irqflags); + + I915_WRITE(DEIMR, c8_regfile->de_imr); + I915_WRITE(DEIER, c8_regfile->de_ier); + I915_WRITE(AUDIMR, c8_regfile->aud_imr); + I915_WRITE(AUDIER, c8_regfile->aud_ier); + I915_WRITE(GTIMR, c8_regfile->gt_imr); + I915_WRITE(GTIER, c8_regfile->gt_ier); + I915_WRITE(GEN6_PMIMR, c8_regfile->pm_imr); + I915_WRITE(GEN6_PMIER, c8_regfile->pm_ier); + I915_WRITE(SRDIMR, c8_regfile->srd_imr); + I915_WRITE(DIGITAL_PORT_HOTPLUG_CNTRL, c8_regfile->hotplug_ctl); + I915_WRITE(GEN7_ERR_INT, c8_regfile->err_int); + + I915_WRITE(SDEIMR, c8_regfile->sde_imr); + I915_WRITE(SDEIER, c8_regfile->sde_ier); + I915_WRITE(_FDI_RXA_IMR, c8_regfile->fdirx_imr); + I915_WRITE(PCH_GTCIMR, c8_regfile->gtcpch_imr); + I915_WRITE(PCH_PORT_HOTPLUG, c8_regfile->shotplug_ctl); + I915_WRITE(SERR_INT, c8_regfile->serr_int); + + spin_unlock_irqrestore(&dev_priv->irq_lock, irqflags); +} + +/* + * This function should only be called by: + * - hsw_set_package_c8 + * - hsw_package_c8_sleep + */ +static void hsw_allow_package_c8(struct drm_i915_private *dev_priv) +{ + uint32_t val; + + WARN_ON(!mutex_is_locked(&dev_priv->c8_lock)); + + /* Disable SCLKGATE_DIS workaround. + * TODO: we should keep this always disabled and only enable it when + * needed. */ + if (dev_priv->pch_id == INTEL_PCH_LPT_LP_DEVICE_ID_TYPE) { + val = I915_READ(SOUTH_DSPCLK_GATE_D); + val &= ~PCH_LP_PARTITION_LEVEL_DISABLE; + I915_WRITE(SOUTH_DSPCLK_GATE_D, val); + } + + lpt_disable_clkout_dp(dev_priv); + hsw_disable_interrupts(dev_priv); + hsw_disable_lcpll(dev_priv); +} + +/* + * This function should only be called by: + * - hsw_set_package_c8 + * - hsw_package_c8_wakeup + */ +static void hsw_disallow_package_c8(struct drm_i915_private *dev_priv) +{ + uint32_t val; + + WARN_ON(!mutex_is_locked(&dev_priv->c8_lock)); + + hsw_restore_lcpll(dev_priv); + hsw_restore_interrupts(dev_priv); + lpt_enable_clkout_dp(dev_priv); + + /* Enable SCLKGATE_DIS workaround. + * TODO: we should keep this always disabled and only enable it when + * needed. */ + if (dev_priv->pch_id == INTEL_PCH_LPT_LP_DEVICE_ID_TYPE) { + val = I915_READ(SOUTH_DSPCLK_GATE_D); + val |= PCH_LP_PARTITION_LEVEL_DISABLE; + I915_WRITE(SOUTH_DSPCLK_GATE_D, val); + } +} + +static bool hsw_can_allow_package_c8(struct drm_i915_private *dev_priv) +{ + struct drm_device *dev = dev_priv->dev; + struct intel_crtc *crtc; + struct intel_encoder *encoder; + uint32_t val; + int used_crtcs = 0, used_encoders = 0; + + list_for_each_entry(crtc, &dev->mode_config.crtc_list, base.head) + if (crtc->base.enabled) + used_crtcs++; + + list_for_each_entry(encoder, &dev->mode_config.encoder_list, + base.head) + if (encoder->connectors_active) + used_encoders++; + + if (used_crtcs || used_encoders) { + DRM_DEBUG_KMS("Not allowing C8: %d crtcs and %d encoders enabled\n", + used_crtcs, used_encoders); + return false; + } + + val = I915_READ(SPLL_CTL); + if (val & SPLL_PLL_ENABLE) { + DRM_DEBUG_KMS("Not allowing C8: SPLL enabled\n"); + return false; + } + + val = I915_READ(WRPLL_CTL1); + if (val & WRPLL_PLL_ENABLE) { + DRM_DEBUG_KMS("Not allowing C8: WRPLL1 enabled\n"); + return false; + } + + val = I915_READ(WRPLL_CTL2); + if (val & WRPLL_PLL_ENABLE) { + DRM_DEBUG_KMS("Not allowing C8: WRPLL2 enabled\n"); + return false; + } + + val = I915_READ(HSW_PWR_WELL_DRIVER); + if (val != 0) { + DRM_DEBUG_KMS("Not allowing C8: power well on\n"); + return false; + } + + val = I915_READ(PCH_PP_STATUS); + if (val & PP_ON) { + DRM_DEBUG_KMS("Not allowing C8: panel power on\n"); + return false; + } + + val = I915_READ(BLC_PWM_CPU_CTL2); + if (val & BLM_PWM_ENABLE) { + DRM_DEBUG_KMS("Not allowing C8: CPU PWM1 enabled\n"); + return false; + } + + val = I915_READ(HSW_BLC_PWM2_CTL); + if (val & BLM_PWM_ENABLE) { + DRM_DEBUG_KMS("Not allowing C8: CPU PWM2 enabled\n"); + return false; + } + + val = I915_READ(BLC_PWM_PCH_CTL1); + if (val & BLM_PCH_PWM_ENABLE) { + DRM_DEBUG_KMS("Not allowing C8: PCH PWM1 enabled\n"); + return false; + } + + val = I915_READ(UTIL_PIN_CTL); + if (val & UTIL_PIN_ENABLE) { + DRM_DEBUG_KMS("Not allowing C8: utility pin enabled\n"); + return false; + } + + val = I915_READ(PCH_GTC_CTL); + if (val & PCH_GTC_ENABLE) { + DRM_DEBUG_KMS("Not allowing C8: PCH GTC enabled\n"); + return false; + } + + DRM_DEBUG_KMS("Allowing package C8+\n"); + return true; +} + +void hsw_package_c8_wakeup(struct drm_i915_private *dev_priv) +{ + if (!dev_priv->allowing_package_c8) + return; + + mutex_lock(&dev_priv->c8_lock); + + dev_priv->c8_wakeup_refcnt++; + + if (dev_priv->c8_wakeup_refcnt == 1) { + DRM_DEBUG_KMS("Waking up from C8\n"); + hsw_disallow_package_c8(dev_priv); + } + + mutex_unlock(&dev_priv->c8_lock); +} + +void hsw_package_c8_sleep(struct drm_i915_private *dev_priv) +{ + if (!dev_priv->allowing_package_c8) + return; + + mutex_lock(&dev_priv->c8_lock); + + dev_priv->c8_wakeup_refcnt--; + + if (dev_priv->c8_wakeup_refcnt == 0) { + DRM_DEBUG_KMS("Sleeping back to C8\n"); + hsw_allow_package_c8(dev_priv); + } + + mutex_unlock(&dev_priv->c8_lock); +} + +static void hsw_set_package_c8(struct drm_i915_private *dev_priv) +{ + bool allow = hsw_can_allow_package_c8(dev_priv); + + mutex_lock(&dev_priv->c8_lock); + + if (allow & !dev_priv->allowing_package_c8) { + hsw_allow_package_c8(dev_priv); + dev_priv->allowing_package_c8 = true; + dev_priv->c8_wakeup_refcnt = 0; + } else if (!allow & dev_priv->allowing_package_c8) { + hsw_disallow_package_c8(dev_priv); + dev_priv->allowing_package_c8 = false; + WARN_ON(dev_priv->c8_wakeup_refcnt); + } + + mutex_unlock(&dev_priv->c8_lock); +} + static void haswell_modeset_global_resources(struct drm_device *dev) { struct drm_i915_private *dev_priv = dev->dev_private; @@ -5778,6 +6235,8 @@ static void haswell_modeset_global_resources(struct drm_device *dev) enable = true; intel_set_power_well(dev, enable); + + hsw_set_package_c8(dev_priv); } static int haswell_crtc_mode_set(struct drm_crtc *crtc, diff --git a/drivers/gpu/drm/i915/intel_dp.c b/drivers/gpu/drm/i915/intel_dp.c index 8845e82..033b45e 100644 --- a/drivers/gpu/drm/i915/intel_dp.c +++ b/drivers/gpu/drm/i915/intel_dp.c @@ -2318,9 +2318,12 @@ intel_dp_detect(struct drm_connector *connector, bool force) struct intel_digital_port *intel_dig_port = dp_to_dig_port(intel_dp); struct intel_encoder *intel_encoder = &intel_dig_port->base; struct drm_device *dev = connector->dev; + struct drm_i915_private *dev_priv = dev->dev_private; enum drm_connector_status status; struct edid *edid = NULL; + hsw_package_c8_wakeup(dev_priv); + intel_dp->has_audio = false; if (HAS_PCH_SPLIT(dev)) @@ -2329,7 +2332,7 @@ intel_dp_detect(struct drm_connector *connector, bool force) status = g4x_dp_detect(intel_dp); if (status != connector_status_connected) - return status; + goto out; intel_dp_probe_oui(intel_dp); @@ -2345,7 +2348,11 @@ intel_dp_detect(struct drm_connector *connector, bool force) if (intel_encoder->type != INTEL_OUTPUT_EDP) intel_encoder->type = INTEL_OUTPUT_DISPLAYPORT; - return connector_status_connected; + status = connector_status_connected; + +out: + hsw_package_c8_sleep(dev_priv); + return status; } static int intel_dp_get_modes(struct drm_connector *connector) diff --git a/drivers/gpu/drm/i915/intel_drv.h b/drivers/gpu/drm/i915/intel_drv.h index 58ff2bf..2b5a352 100644 --- a/drivers/gpu/drm/i915/intel_drv.h +++ b/drivers/gpu/drm/i915/intel_drv.h @@ -698,6 +698,8 @@ extern void intel_update_fbc(struct drm_device *dev); extern void intel_gpu_ips_init(struct drm_i915_private *dev_priv); extern void intel_gpu_ips_teardown(void); +extern void hsw_package_c8_wakeup(struct drm_i915_private *dev_priv); +extern void hsw_package_c8_sleep(struct drm_i915_private *dev_priv); extern bool intel_using_power_well(struct drm_device *dev); extern void intel_init_power_well(struct drm_device *dev); extern void intel_set_power_well(struct drm_device *dev, bool enable); diff --git a/drivers/gpu/drm/i915/intel_hdmi.c b/drivers/gpu/drm/i915/intel_hdmi.c index ee4a8da..6531158b 100644 --- a/drivers/gpu/drm/i915/intel_hdmi.c +++ b/drivers/gpu/drm/i915/intel_hdmi.c @@ -819,6 +819,8 @@ intel_hdmi_detect(struct drm_connector *connector, bool force) struct edid *edid; enum drm_connector_status status = connector_status_disconnected; + hsw_package_c8_wakeup(dev_priv); + intel_hdmi->has_hdmi_sink = false; intel_hdmi->has_audio = false; intel_hdmi->rgb_quant_range_selectable = false; @@ -846,6 +848,7 @@ intel_hdmi_detect(struct drm_connector *connector, bool force) intel_encoder->type = INTEL_OUTPUT_HDMI; } + hsw_package_c8_sleep(dev_priv); return status; } -- 1.7.10.4