drm/i915/tc: Check the PLL type used by an enabled TC port
authorImre Deak <imre.deak@intel.com>
Tue, 21 Mar 2023 22:01:01 +0000 (00:01 +0200)
committerImre Deak <imre.deak@intel.com>
Wed, 22 Mar 2023 18:31:23 +0000 (20:31 +0200)
The current way to determine during HW state sanitization if a PHY is
connected in the expected way doesn't work in all cases. The check for
this considers only the PHY ready/owned state and the initial TC mode
which was determined earlier by the TC port HW readout - using the
sink's HPD and the same PHY ready/owned states.

For instance for an enabled DP-alt/TBT port without the PHY ready/owned
flags set the initial mode will be TBT, and this will be regarded as a
valid PHY state. However it's possible that the port is actually enabled
in DP-alt mode, but for some reason the PHY ownership was not acquired.

Make sure the driver can detect invalid PHY states as in the above
example by checking the PHY ready/owned state wrt. the PLL type used.
This should be the TBT PLL if the PHY is not owned and the MG (non-TBT)
PLL if the PHY is owned.

v2: Rebased on change passing crtc_state in the previous patch.

Reviewed-by: Ville Syrjälä <ville.syrjala@linux.intel.com>
Signed-off-by: Imre Deak <imre.deak@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20230321220101.983366-3-imre.deak@intel.com
drivers/gpu/drm/i915/display/intel_tc.c

index c5bfd9f11d7de9f5233a7219022dfbf0f1d02cab..bd8c9df5f98fedac98d217de83db5198986ff0c2 100644 (file)
@@ -5,6 +5,7 @@
 
 #include "i915_drv.h"
 #include "i915_reg.h"
+#include "intel_ddi.h"
 #include "intel_de.h"
 #include "intel_display.h"
 #include "intel_display_power_map.h"
@@ -568,29 +569,29 @@ static bool tc_phy_is_ready_and_owned(struct intel_digital_port *dig_port,
        return phy_is_ready && phy_is_owned;
 }
 
-static bool icl_tc_phy_is_connected(struct intel_digital_port *dig_port)
+static bool tc_phy_is_connected(struct intel_digital_port *dig_port,
+                               enum icl_port_dpll_id port_pll_type)
 {
-       struct drm_i915_private *i915 = to_i915(dig_port->base.base.dev);
-
-       if (!tc_phy_status_complete(dig_port)) {
-               drm_dbg_kms(&i915->drm, "Port %s: PHY status not complete\n",
-                           dig_port->tc_port_name);
-               return dig_port->tc_mode == TC_PORT_TBT_ALT;
-       }
-
-       /* On ADL-P the PHY complete flag is set in TBT mode as well. */
-       if (IS_ALDERLAKE_P(i915) && dig_port->tc_mode == TC_PORT_TBT_ALT)
-               return true;
+       struct intel_encoder *encoder = &dig_port->base;
+       struct drm_i915_private *i915 = to_i915(encoder->base.dev);
+       bool phy_is_ready = tc_phy_status_complete(dig_port);
+       bool phy_is_owned = tc_phy_is_owned(dig_port);
+       bool is_connected;
 
-       if (!tc_phy_is_owned(dig_port)) {
-               drm_dbg_kms(&i915->drm, "Port %s: PHY not owned\n",
-                           dig_port->tc_port_name);
+       if (tc_phy_is_ready_and_owned(dig_port, phy_is_ready, phy_is_owned))
+               is_connected = port_pll_type == ICL_PORT_DPLL_MG_PHY;
+       else
+               is_connected = port_pll_type == ICL_PORT_DPLL_DEFAULT;
 
-               return false;
-       }
+       drm_dbg_kms(&i915->drm,
+                   "Port %s: PHY connected: %s (ready: %s, owned: %s, pll_type: %s)\n",
+                   dig_port->tc_port_name,
+                   str_yes_no(is_connected),
+                   str_yes_no(phy_is_ready),
+                   str_yes_no(phy_is_owned),
+                   port_pll_type == ICL_PORT_DPLL_DEFAULT ? "tbt" : "non-tbt");
 
-       return dig_port->tc_mode == TC_PORT_DP_ALT ||
-              dig_port->tc_mode == TC_PORT_LEGACY;
+       return is_connected;
 }
 
 static void tc_phy_wait_for_ready(struct intel_digital_port *dig_port)
@@ -876,15 +877,18 @@ static bool tc_port_has_active_links(struct intel_digital_port *dig_port,
                                     const struct intel_crtc_state *crtc_state)
 {
        struct drm_i915_private *i915 = to_i915(dig_port->base.base.dev);
+       enum icl_port_dpll_id pll_type = ICL_PORT_DPLL_DEFAULT;
        int active_links = 0;
 
        if (dig_port->dp.is_mst) {
+               /* TODO: get the PLL type for MST, once HW readout is done for it. */
                active_links = intel_dp_mst_encoder_active_links(dig_port);
        } else if (crtc_state && crtc_state->hw.active) {
+               pll_type = intel_ddi_port_pll_type(&dig_port->base, crtc_state);
                active_links = 1;
        }
 
-       if (active_links && !icl_tc_phy_is_connected(dig_port))
+       if (active_links && !tc_phy_is_connected(dig_port, pll_type))
                drm_err(&i915->drm,
                        "Port %s: PHY disconnected with %d active link(s)\n",
                        dig_port->tc_port_name, active_links);