|
@@ -7149,7 +7149,8 @@ static void intel_pch_transcoder_get_m_n(struct intel_crtc *crtc,
|
|
|
|
|
|
static void intel_cpu_transcoder_get_m_n(struct intel_crtc *crtc,
|
|
|
enum transcoder transcoder,
|
|
|
- struct intel_link_m_n *m_n)
|
|
|
+ struct intel_link_m_n *m_n,
|
|
|
+ struct intel_link_m_n *m2_n2)
|
|
|
{
|
|
|
struct drm_device *dev = crtc->base.dev;
|
|
|
struct drm_i915_private *dev_priv = dev->dev_private;
|
|
@@ -7163,6 +7164,20 @@ static void intel_cpu_transcoder_get_m_n(struct intel_crtc *crtc,
|
|
|
m_n->gmch_n = I915_READ(PIPE_DATA_N1(transcoder));
|
|
|
m_n->tu = ((I915_READ(PIPE_DATA_M1(transcoder))
|
|
|
& TU_SIZE_MASK) >> TU_SIZE_SHIFT) + 1;
|
|
|
+ /* Read M2_N2 registers only for gen < 8 (M2_N2 available for
|
|
|
+ * gen < 8) and if DRRS is supported (to make sure the
|
|
|
+ * registers are not unnecessarily read).
|
|
|
+ */
|
|
|
+ if (m2_n2 && INTEL_INFO(dev)->gen < 8 &&
|
|
|
+ crtc->config.has_drrs) {
|
|
|
+ m2_n2->link_m = I915_READ(PIPE_LINK_M2(transcoder));
|
|
|
+ m2_n2->link_n = I915_READ(PIPE_LINK_N2(transcoder));
|
|
|
+ m2_n2->gmch_m = I915_READ(PIPE_DATA_M2(transcoder))
|
|
|
+ & ~TU_SIZE_MASK;
|
|
|
+ m2_n2->gmch_n = I915_READ(PIPE_DATA_N2(transcoder));
|
|
|
+ m2_n2->tu = ((I915_READ(PIPE_DATA_M2(transcoder))
|
|
|
+ & TU_SIZE_MASK) >> TU_SIZE_SHIFT) + 1;
|
|
|
+ }
|
|
|
} else {
|
|
|
m_n->link_m = I915_READ(PIPE_LINK_M_G4X(pipe));
|
|
|
m_n->link_n = I915_READ(PIPE_LINK_N_G4X(pipe));
|
|
@@ -7181,14 +7196,15 @@ void intel_dp_get_m_n(struct intel_crtc *crtc,
|
|
|
intel_pch_transcoder_get_m_n(crtc, &pipe_config->dp_m_n);
|
|
|
else
|
|
|
intel_cpu_transcoder_get_m_n(crtc, pipe_config->cpu_transcoder,
|
|
|
- &pipe_config->dp_m_n);
|
|
|
+ &pipe_config->dp_m_n,
|
|
|
+ &pipe_config->dp_m2_n2);
|
|
|
}
|
|
|
|
|
|
static void ironlake_get_fdi_m_n_config(struct intel_crtc *crtc,
|
|
|
struct intel_crtc_config *pipe_config)
|
|
|
{
|
|
|
intel_cpu_transcoder_get_m_n(crtc, pipe_config->cpu_transcoder,
|
|
|
- &pipe_config->fdi_m_n);
|
|
|
+ &pipe_config->fdi_m_n, NULL);
|
|
|
}
|
|
|
|
|
|
static void ironlake_get_pfit_config(struct intel_crtc *crtc,
|
|
@@ -10005,6 +10021,15 @@ static void intel_dump_pipe_config(struct intel_crtc *crtc,
|
|
|
pipe_config->dp_m_n.gmch_m, pipe_config->dp_m_n.gmch_n,
|
|
|
pipe_config->dp_m_n.link_m, pipe_config->dp_m_n.link_n,
|
|
|
pipe_config->dp_m_n.tu);
|
|
|
+
|
|
|
+ DRM_DEBUG_KMS("dp: %i, gmch_m2: %u, gmch_n2: %u, link_m2: %u, link_n2: %u, tu2: %u\n",
|
|
|
+ pipe_config->has_dp_encoder,
|
|
|
+ pipe_config->dp_m2_n2.gmch_m,
|
|
|
+ pipe_config->dp_m2_n2.gmch_n,
|
|
|
+ pipe_config->dp_m2_n2.link_m,
|
|
|
+ pipe_config->dp_m2_n2.link_n,
|
|
|
+ pipe_config->dp_m2_n2.tu);
|
|
|
+
|
|
|
DRM_DEBUG_KMS("requested mode:\n");
|
|
|
drm_mode_debug_printmodeline(&pipe_config->requested_mode);
|
|
|
DRM_DEBUG_KMS("adjusted mode:\n");
|
|
@@ -10385,6 +10410,22 @@ intel_pipe_config_compare(struct drm_device *dev,
|
|
|
return false; \
|
|
|
}
|
|
|
|
|
|
+/* This is required for BDW+ where there is only one set of registers for
|
|
|
+ * switching between high and low RR.
|
|
|
+ * This macro can be used whenever a comparison has to be made between one
|
|
|
+ * hw state and multiple sw state variables.
|
|
|
+ */
|
|
|
+#define PIPE_CONF_CHECK_I_ALT(name, alt_name) \
|
|
|
+ if ((current_config->name != pipe_config->name) && \
|
|
|
+ (current_config->alt_name != pipe_config->name)) { \
|
|
|
+ DRM_ERROR("mismatch in " #name " " \
|
|
|
+ "(expected %i or %i, found %i)\n", \
|
|
|
+ current_config->name, \
|
|
|
+ current_config->alt_name, \
|
|
|
+ pipe_config->name); \
|
|
|
+ return false; \
|
|
|
+ }
|
|
|
+
|
|
|
#define PIPE_CONF_CHECK_FLAGS(name, mask) \
|
|
|
if ((current_config->name ^ pipe_config->name) & (mask)) { \
|
|
|
DRM_ERROR("mismatch in " #name "(" #mask ") " \
|
|
@@ -10417,11 +10458,28 @@ intel_pipe_config_compare(struct drm_device *dev,
|
|
|
PIPE_CONF_CHECK_I(fdi_m_n.tu);
|
|
|
|
|
|
PIPE_CONF_CHECK_I(has_dp_encoder);
|
|
|
- PIPE_CONF_CHECK_I(dp_m_n.gmch_m);
|
|
|
- PIPE_CONF_CHECK_I(dp_m_n.gmch_n);
|
|
|
- PIPE_CONF_CHECK_I(dp_m_n.link_m);
|
|
|
- PIPE_CONF_CHECK_I(dp_m_n.link_n);
|
|
|
- PIPE_CONF_CHECK_I(dp_m_n.tu);
|
|
|
+
|
|
|
+ if (INTEL_INFO(dev)->gen < 8) {
|
|
|
+ PIPE_CONF_CHECK_I(dp_m_n.gmch_m);
|
|
|
+ PIPE_CONF_CHECK_I(dp_m_n.gmch_n);
|
|
|
+ PIPE_CONF_CHECK_I(dp_m_n.link_m);
|
|
|
+ PIPE_CONF_CHECK_I(dp_m_n.link_n);
|
|
|
+ PIPE_CONF_CHECK_I(dp_m_n.tu);
|
|
|
+
|
|
|
+ if (current_config->has_drrs) {
|
|
|
+ PIPE_CONF_CHECK_I(dp_m2_n2.gmch_m);
|
|
|
+ PIPE_CONF_CHECK_I(dp_m2_n2.gmch_n);
|
|
|
+ PIPE_CONF_CHECK_I(dp_m2_n2.link_m);
|
|
|
+ PIPE_CONF_CHECK_I(dp_m2_n2.link_n);
|
|
|
+ PIPE_CONF_CHECK_I(dp_m2_n2.tu);
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ PIPE_CONF_CHECK_I_ALT(dp_m_n.gmch_m, dp_m2_n2.gmch_m);
|
|
|
+ PIPE_CONF_CHECK_I_ALT(dp_m_n.gmch_n, dp_m2_n2.gmch_n);
|
|
|
+ PIPE_CONF_CHECK_I_ALT(dp_m_n.link_m, dp_m2_n2.link_m);
|
|
|
+ PIPE_CONF_CHECK_I_ALT(dp_m_n.link_n, dp_m2_n2.link_n);
|
|
|
+ PIPE_CONF_CHECK_I_ALT(dp_m_n.tu, dp_m2_n2.tu);
|
|
|
+ }
|
|
|
|
|
|
PIPE_CONF_CHECK_I(adjusted_mode.crtc_hdisplay);
|
|
|
PIPE_CONF_CHECK_I(adjusted_mode.crtc_htotal);
|
|
@@ -10507,6 +10565,7 @@ intel_pipe_config_compare(struct drm_device *dev,
|
|
|
|
|
|
#undef PIPE_CONF_CHECK_X
|
|
|
#undef PIPE_CONF_CHECK_I
|
|
|
+#undef PIPE_CONF_CHECK_I_ALT
|
|
|
#undef PIPE_CONF_CHECK_FLAGS
|
|
|
#undef PIPE_CONF_CHECK_CLOCK_FUZZY
|
|
|
#undef PIPE_CONF_QUIRK
|