]> www.infradead.org Git - users/jedix/linux-maple.git/commitdiff
drm/i915/lnl+/tc: Fix handling of an enabled/disconnected dp-alt sink
authorImre Deak <imre.deak@intel.com>
Mon, 11 Aug 2025 08:01:48 +0000 (11:01 +0300)
committerTvrtko Ursulin <tursulin@ursulin.net>
Mon, 18 Aug 2025 07:08:20 +0000 (08:08 +0100)
The TypeC PHY HW readout during driver loading and system resume
determines which TypeC mode the PHY is in (legacy/DP-alt/TBT-alt) and
whether the PHY is connected, based on the PHY's Owned and Ready flags.
For the PHY to be in DP-alt or legacy mode and for the PHY to be in the
connected state in these modes, both the Owned (set by the BIOS/driver)
and the Ready (set by the HW) flags should be set.

On ICL-MTL the HW kept the PHY's Ready flag set after the driver
connected the PHY by acquiring the PHY ownership (by setting the Owned
flag), until the driver disconnected the PHY by releasing the PHY
ownership (by clearing the Owned flag). On LNL+ this has changed, in
that the HW clears the Ready flag as soon as the sink gets disconnected,
even if the PHY ownership was acquired already and hence the PHY is
being used by the display.

When inheriting the HW state from BIOS for a PHY connected in DP-alt
mode on which the sink got disconnected - i.e. in a case where the sink
was connected while BIOS/GOP was running and so the sink got enabled
connecting the PHY, but the user disconnected the sink by the time the
driver loaded - the PHY Owned but not Ready state must be accounted for
on LNL+ according to the above. Do that by assuming on LNL+ that the PHY
is connected in DP-alt mode whenever the PHY Owned flag is set,
regardless of the PHY Ready flag.

This fixes a problem on LNL+, where the PHY TypeC mode / connected state
was detected incorrectly for a DP-alt sink, which got connected and then
disconnected by the user in the above way.

v2: Rename tc_phy_in_legacy_or_dp_alt_mode() to tc_phy_owned_by_display().
    (Luca, Jani)

Cc: Jani Nikula <jani.nikula@intel.com>
Cc: stable@vger.kernel.org # v6.8+
Reported-by: Charlton Lin <charlton.lin@intel.com>
Tested-by: Khaled Almahallawy <khaled.almahallawy@intel.com>
Reviewed-by: Mika Kahola <mika.kahola@intel.com>
Reviewed-by: Luca Coelho <luciano.coelho@intel.com>
[Imre: Add one-liner function documentation for tc_phy_owned_by_display()]
Signed-off-by: Imre Deak <imre.deak@intel.com>
Link: https://lore.kernel.org/r/20250811080152.906216-2-imre.deak@intel.com
(cherry picked from commit 89f4b196ee4b056e0e8c179b247b29d4a71a4e7e)
Signed-off-by: Tvrtko Ursulin <tursulin@ursulin.net>
drivers/gpu/drm/i915/display/intel_tc.c

index 3bc57579fe53e86911b2e1f8fdd158a73ab3b748..8208539bfe66ef3691f5d10e136475f587e5d98e 100644 (file)
@@ -1226,14 +1226,19 @@ static void tc_phy_get_hw_state(struct intel_tc_port *tc)
        tc->phy_ops->get_hw_state(tc);
 }
 
-static bool tc_phy_is_ready_and_owned(struct intel_tc_port *tc,
-                                     bool phy_is_ready, bool phy_is_owned)
+/* Is the PHY owned by display i.e. is it in legacy or DP-alt mode? */
+static bool tc_phy_owned_by_display(struct intel_tc_port *tc,
+                                   bool phy_is_ready, bool phy_is_owned)
 {
        struct intel_display *display = to_intel_display(tc->dig_port);
 
-       drm_WARN_ON(display->drm, phy_is_owned && !phy_is_ready);
+       if (DISPLAY_VER(display) < 20) {
+               drm_WARN_ON(display->drm, phy_is_owned && !phy_is_ready);
 
-       return phy_is_ready && phy_is_owned;
+               return phy_is_ready && phy_is_owned;
+       } else {
+               return phy_is_owned;
+       }
 }
 
 static bool tc_phy_is_connected(struct intel_tc_port *tc,
@@ -1244,7 +1249,7 @@ static bool tc_phy_is_connected(struct intel_tc_port *tc,
        bool phy_is_owned = tc_phy_is_owned(tc);
        bool is_connected;
 
-       if (tc_phy_is_ready_and_owned(tc, phy_is_ready, phy_is_owned))
+       if (tc_phy_owned_by_display(tc, 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;
@@ -1352,7 +1357,7 @@ tc_phy_get_current_mode(struct intel_tc_port *tc)
        phy_is_ready = tc_phy_is_ready(tc);
        phy_is_owned = tc_phy_is_owned(tc);
 
-       if (!tc_phy_is_ready_and_owned(tc, phy_is_ready, phy_is_owned)) {
+       if (!tc_phy_owned_by_display(tc, phy_is_ready, phy_is_owned)) {
                mode = get_tc_mode_in_phy_not_owned_state(tc, live_mode);
        } else {
                drm_WARN_ON(display->drm, live_mode == TC_PORT_TBT_ALT);