diff options
Diffstat (limited to 'drivers/gpu/drm/i915/display/intel_dp.c')
| -rw-r--r-- | drivers/gpu/drm/i915/display/intel_dp.c | 292 | 
1 files changed, 45 insertions, 247 deletions
| diff --git a/drivers/gpu/drm/i915/display/intel_dp.c b/drivers/gpu/drm/i915/display/intel_dp.c index d0fc34826771..0eb5d66f87a7 100644 --- a/drivers/gpu/drm/i915/display/intel_dp.c +++ b/drivers/gpu/drm/i915/display/intel_dp.c @@ -62,6 +62,7 @@  #include "intel_panel.h"  #include "intel_psr.h"  #include "intel_sideband.h" +#include "intel_tc.h"  #include "intel_vdsc.h"  #define DP_DPRX_ESI_LEN 14 @@ -211,47 +212,13 @@ static int intel_dp_max_common_rate(struct intel_dp *intel_dp)  	return intel_dp->common_rates[intel_dp->num_common_rates - 1];  } -static int intel_dp_get_fia_supported_lane_count(struct intel_dp *intel_dp) -{ -	struct intel_digital_port *dig_port = dp_to_dig_port(intel_dp); -	struct drm_i915_private *dev_priv = to_i915(dig_port->base.base.dev); -	enum tc_port tc_port = intel_port_to_tc(dev_priv, dig_port->base.port); -	intel_wakeref_t wakeref; -	u32 lane_info; - -	if (tc_port == PORT_TC_NONE || dig_port->tc_type != TC_PORT_TYPEC) -		return 4; - -	lane_info = 0; -	with_intel_display_power(dev_priv, POWER_DOMAIN_DISPLAY_CORE, wakeref) -		lane_info = (I915_READ(PORT_TX_DFLEXDPSP) & -			     DP_LANE_ASSIGNMENT_MASK(tc_port)) >> -				DP_LANE_ASSIGNMENT_SHIFT(tc_port); - -	switch (lane_info) { -	default: -		MISSING_CASE(lane_info); -		/* fall through */ -	case 1: -	case 2: -	case 4: -	case 8: -		return 1; -	case 3: -	case 12: -		return 2; -	case 15: -		return 4; -	} -} -  /* Theoretical max between source and sink */  static int intel_dp_max_common_lane_count(struct intel_dp *intel_dp)  {  	struct intel_digital_port *intel_dig_port = dp_to_dig_port(intel_dp);  	int source_max = intel_dig_port->max_lanes;  	int sink_max = drm_dp_max_lane_count(intel_dp->dpcd); -	int fia_max = intel_dp_get_fia_supported_lane_count(intel_dp); +	int fia_max = intel_tc_port_fia_max_lane_count(intel_dig_port);  	return min3(source_max, sink_max, fia_max);  } @@ -330,9 +297,9 @@ static int icl_max_source_rate(struct intel_dp *intel_dp)  {  	struct intel_digital_port *dig_port = dp_to_dig_port(intel_dp);  	struct drm_i915_private *dev_priv = to_i915(dig_port->base.base.dev); -	enum port port = dig_port->base.port; +	enum phy phy = intel_port_to_phy(dev_priv, dig_port->base.port); -	if (intel_port_is_combophy(dev_priv, port) && +	if (intel_phy_is_combo(dev_priv, phy) &&  	    !IS_ELKHARTLAKE(dev_priv) &&  	    !intel_dp_is_edp(intel_dp))  		return 540000; @@ -1209,7 +1176,7 @@ static u32 skl_get_aux_send_ctl(struct intel_dp *intel_dp,  	      DP_AUX_CH_CTL_FW_SYNC_PULSE_SKL(32) |  	      DP_AUX_CH_CTL_SYNC_PULSE_SKL(32); -	if (intel_dig_port->tc_type == TC_PORT_TBT) +	if (intel_dig_port->tc_mode == TC_PORT_TBT_ALT)  		ret |= DP_AUX_CH_CTL_TBT_IO;  	return ret; @@ -1225,6 +1192,8 @@ intel_dp_aux_xfer(struct intel_dp *intel_dp,  	struct drm_i915_private *i915 =  			to_i915(intel_dig_port->base.base.dev);  	struct intel_uncore *uncore = &i915->uncore; +	enum phy phy = intel_port_to_phy(i915, intel_dig_port->base.port); +	bool is_tc_port = intel_phy_is_tc(i915, phy);  	i915_reg_t ch_ctl, ch_data[5];  	u32 aux_clock_divider;  	enum intel_display_power_domain aux_domain = @@ -1240,6 +1209,9 @@ intel_dp_aux_xfer(struct intel_dp *intel_dp,  	for (i = 0; i < ARRAY_SIZE(ch_data); i++)  		ch_data[i] = intel_dp->aux_ch_data_reg(intel_dp, i); +	if (is_tc_port) +		intel_tc_port_lock(intel_dig_port); +  	aux_wakeref = intel_display_power_get(i915, aux_domain);  	pps_wakeref = pps_lock(intel_dp); @@ -1392,6 +1364,9 @@ out:  	pps_unlock(intel_dp, pps_wakeref);  	intel_display_power_put_async(i915, aux_domain, aux_wakeref); +	if (is_tc_port) +		intel_tc_port_unlock(intel_dig_port); +  	return ret;  } @@ -1879,8 +1854,10 @@ intel_dp_compute_link_config_wide(struct intel_dp *intel_dp,  	int mode_rate, link_clock, link_avail;  	for (bpp = limits->max_bpp; bpp >= limits->min_bpp; bpp -= 2 * 3) { +		int output_bpp = intel_dp_output_bpp(pipe_config, bpp); +  		mode_rate = intel_dp_link_required(adjusted_mode->crtc_clock, -						   bpp); +						   output_bpp);  		for (clock = limits->min_clock; clock <= limits->max_clock; clock++) {  			for (lane_count = limits->min_lane_count; @@ -4244,8 +4221,14 @@ intel_dp_get_dpcd(struct intel_dp *intel_dp)  	if (!intel_dp_read_dpcd(intel_dp))  		return false; -	/* Don't clobber cached eDP rates. */ +	/* +	 * Don't clobber cached eDP rates. Also skip re-reading +	 * the OUI/ID since we know it won't change. +	 */  	if (!intel_dp_is_edp(intel_dp)) { +		drm_dp_read_desc(&intel_dp->aux, &intel_dp->desc, +				 drm_dp_is_branch(intel_dp->dpcd)); +  		intel_dp_set_sink_rates(intel_dp);  		intel_dp_set_common_rates(intel_dp);  	} @@ -4254,7 +4237,8 @@ intel_dp_get_dpcd(struct intel_dp *intel_dp)  	 * Some eDP panels do not set a valid value for sink count, that is why  	 * it don't care about read it here and in intel_edp_init_dpcd().  	 */ -	if (!intel_dp_is_edp(intel_dp)) { +	if (!intel_dp_is_edp(intel_dp) && +	    !drm_dp_has_quirk(&intel_dp->desc, DP_DPCD_QUIRK_NO_SINK_COUNT)) {  		u8 count;  		ssize_t r; @@ -4879,14 +4863,16 @@ int intel_dp_retrain_link(struct intel_encoder *encoder,   * retrain the link to get a picture. That's in case no   * userspace component reacted to intermittent HPD dip.   */ -static bool intel_dp_hotplug(struct intel_encoder *encoder, -			     struct intel_connector *connector) +static enum intel_hotplug_state +intel_dp_hotplug(struct intel_encoder *encoder, +		 struct intel_connector *connector, +		 bool irq_received)  {  	struct drm_modeset_acquire_ctx ctx; -	bool changed; +	enum intel_hotplug_state state;  	int ret; -	changed = intel_encoder_hotplug(encoder, connector); +	state = intel_encoder_hotplug(encoder, connector, irq_received);  	drm_modeset_acquire_init(&ctx, 0); @@ -4905,7 +4891,14 @@ static bool intel_dp_hotplug(struct intel_encoder *encoder,  	drm_modeset_acquire_fini(&ctx);  	WARN(ret, "Acquiring modeset locks failed with %i\n", ret); -	return changed; +	/* +	 * Keeping it consistent with intel_ddi_hotplug() and +	 * intel_hdmi_hotplug(). +	 */ +	if (state == INTEL_HOTPLUG_UNCHANGED && irq_received) +		state = INTEL_HOTPLUG_RETRY; + +	return state;  }  static void intel_dp_check_service_irq(struct intel_dp *intel_dp) @@ -5233,204 +5226,16 @@ static bool icl_combo_port_connected(struct drm_i915_private *dev_priv,  	return I915_READ(SDEISR) & SDE_DDI_HOTPLUG_ICP(port);  } -static const char *tc_type_name(enum tc_port_type type) -{ -	static const char * const names[] = { -		[TC_PORT_UNKNOWN] = "unknown", -		[TC_PORT_LEGACY] = "legacy", -		[TC_PORT_TYPEC] = "typec", -		[TC_PORT_TBT] = "tbt", -	}; - -	if (WARN_ON(type >= ARRAY_SIZE(names))) -		type = TC_PORT_UNKNOWN; - -	return names[type]; -} - -static void icl_update_tc_port_type(struct drm_i915_private *dev_priv, -				    struct intel_digital_port *intel_dig_port, -				    bool is_legacy, bool is_typec, bool is_tbt) -{ -	enum port port = intel_dig_port->base.port; -	enum tc_port_type old_type = intel_dig_port->tc_type; - -	WARN_ON(is_legacy + is_typec + is_tbt != 1); - -	if (is_legacy) -		intel_dig_port->tc_type = TC_PORT_LEGACY; -	else if (is_typec) -		intel_dig_port->tc_type = TC_PORT_TYPEC; -	else if (is_tbt) -		intel_dig_port->tc_type = TC_PORT_TBT; -	else -		return; - -	/* Types are not supposed to be changed at runtime. */ -	WARN_ON(old_type != TC_PORT_UNKNOWN && -		old_type != intel_dig_port->tc_type); - -	if (old_type != intel_dig_port->tc_type) -		DRM_DEBUG_KMS("Port %c has TC type %s\n", port_name(port), -			      tc_type_name(intel_dig_port->tc_type)); -} - -/* - * This function implements the first part of the Connect Flow described by our - * specification, Gen11 TypeC Programming chapter. The rest of the flow (reading - * lanes, EDID, etc) is done as needed in the typical places. - * - * Unlike the other ports, type-C ports are not available to use as soon as we - * get a hotplug. The type-C PHYs can be shared between multiple controllers: - * display, USB, etc. As a result, handshaking through FIA is required around - * connect and disconnect to cleanly transfer ownership with the controller and - * set the type-C power state. - * - * We could opt to only do the connect flow when we actually try to use the AUX - * channels or do a modeset, then immediately run the disconnect flow after - * usage, but there are some implications on this for a dynamic environment: - * things may go away or change behind our backs. So for now our driver is - * always trying to acquire ownership of the controller as soon as it gets an - * interrupt (or polls state and sees a port is connected) and only gives it - * back when it sees a disconnect. Implementation of a more fine-grained model - * will require a lot of coordination with user space and thorough testing for - * the extra possible cases. - */ -static bool icl_tc_phy_connect(struct drm_i915_private *dev_priv, -			       struct intel_digital_port *dig_port) -{ -	enum tc_port tc_port = intel_port_to_tc(dev_priv, dig_port->base.port); -	u32 val; - -	if (dig_port->tc_type != TC_PORT_LEGACY && -	    dig_port->tc_type != TC_PORT_TYPEC) -		return true; - -	val = I915_READ(PORT_TX_DFLEXDPPMS); -	if (!(val & DP_PHY_MODE_STATUS_COMPLETED(tc_port))) { -		DRM_DEBUG_KMS("DP PHY for TC port %d not ready\n", tc_port); -		WARN_ON(dig_port->tc_legacy_port); -		return false; -	} - -	/* -	 * This function may be called many times in a row without an HPD event -	 * in between, so try to avoid the write when we can. -	 */ -	val = I915_READ(PORT_TX_DFLEXDPCSSS); -	if (!(val & DP_PHY_MODE_STATUS_NOT_SAFE(tc_port))) { -		val |= DP_PHY_MODE_STATUS_NOT_SAFE(tc_port); -		I915_WRITE(PORT_TX_DFLEXDPCSSS, val); -	} - -	/* -	 * Now we have to re-check the live state, in case the port recently -	 * became disconnected. Not necessary for legacy mode. -	 */ -	if (dig_port->tc_type == TC_PORT_TYPEC && -	    !(I915_READ(PORT_TX_DFLEXDPSP) & TC_LIVE_STATE_TC(tc_port))) { -		DRM_DEBUG_KMS("TC PHY %d sudden disconnect.\n", tc_port); -		icl_tc_phy_disconnect(dev_priv, dig_port); -		return false; -	} - -	return true; -} - -/* - * See the comment at the connect function. This implements the Disconnect - * Flow. - */ -void icl_tc_phy_disconnect(struct drm_i915_private *dev_priv, -			   struct intel_digital_port *dig_port) -{ -	enum tc_port tc_port = intel_port_to_tc(dev_priv, dig_port->base.port); - -	if (dig_port->tc_type == TC_PORT_UNKNOWN) -		return; - -	/* -	 * TBT disconnection flow is read the live status, what was done in -	 * caller. -	 */ -	if (dig_port->tc_type == TC_PORT_TYPEC || -	    dig_port->tc_type == TC_PORT_LEGACY) { -		u32 val; - -		val = I915_READ(PORT_TX_DFLEXDPCSSS); -		val &= ~DP_PHY_MODE_STATUS_NOT_SAFE(tc_port); -		I915_WRITE(PORT_TX_DFLEXDPCSSS, val); -	} - -	DRM_DEBUG_KMS("Port %c TC type %s disconnected\n", -		      port_name(dig_port->base.port), -		      tc_type_name(dig_port->tc_type)); - -	dig_port->tc_type = TC_PORT_UNKNOWN; -} - -/* - * The type-C ports are different because even when they are connected, they may - * not be available/usable by the graphics driver: see the comment on - * icl_tc_phy_connect(). So in our driver instead of adding the additional - * concept of "usable" and make everything check for "connected and usable" we - * define a port as "connected" when it is not only connected, but also when it - * is usable by the rest of the driver. That maintains the old assumption that - * connected ports are usable, and avoids exposing to the users objects they - * can't really use. - */ -static bool icl_tc_port_connected(struct drm_i915_private *dev_priv, -				  struct intel_digital_port *intel_dig_port) -{ -	enum port port = intel_dig_port->base.port; -	enum tc_port tc_port = intel_port_to_tc(dev_priv, port); -	bool is_legacy, is_typec, is_tbt; -	u32 dpsp; - -	/* -	 * Complain if we got a legacy port HPD, but VBT didn't mark the port as -	 * legacy. Treat the port as legacy from now on. -	 */ -	if (!intel_dig_port->tc_legacy_port && -	    I915_READ(SDEISR) & SDE_TC_HOTPLUG_ICP(tc_port)) { -		DRM_ERROR("VBT incorrectly claims port %c is not TypeC legacy\n", -			  port_name(port)); -		intel_dig_port->tc_legacy_port = true; -	} -	is_legacy = intel_dig_port->tc_legacy_port; - -	/* -	 * The spec says we shouldn't be using the ISR bits for detecting -	 * between TC and TBT. We should use DFLEXDPSP. -	 */ -	dpsp = I915_READ(PORT_TX_DFLEXDPSP); -	is_typec = dpsp & TC_LIVE_STATE_TC(tc_port); -	is_tbt = dpsp & TC_LIVE_STATE_TBT(tc_port); - -	if (!is_legacy && !is_typec && !is_tbt) { -		icl_tc_phy_disconnect(dev_priv, intel_dig_port); - -		return false; -	} - -	icl_update_tc_port_type(dev_priv, intel_dig_port, is_legacy, is_typec, -				is_tbt); - -	if (!icl_tc_phy_connect(dev_priv, intel_dig_port)) -		return false; - -	return true; -} -  static bool icl_digital_port_connected(struct intel_encoder *encoder)  {  	struct drm_i915_private *dev_priv = to_i915(encoder->base.dev);  	struct intel_digital_port *dig_port = enc_to_dig_port(&encoder->base); +	enum phy phy = intel_port_to_phy(dev_priv, encoder->port); -	if (intel_port_is_combophy(dev_priv, encoder->port)) +	if (intel_phy_is_combo(dev_priv, phy))  		return icl_combo_port_connected(dev_priv, dig_port); -	else if (intel_port_is_tc(dev_priv, encoder->port)) -		return icl_tc_port_connected(dev_priv, dig_port); +	else if (intel_phy_is_tc(dev_priv, phy)) +		return intel_tc_port_connected(dig_port);  	else  		MISSING_CASE(encoder->hpd_pin); @@ -5588,9 +5393,6 @@ intel_dp_detect(struct drm_connector *connector,  	if (INTEL_GEN(dev_priv) >= 11)  		intel_dp_get_dsc_sink_cap(intel_dp); -	drm_dp_read_desc(&intel_dp->aux, &intel_dp->desc, -			 drm_dp_is_branch(intel_dp->dpcd)); -  	intel_dp_configure_mst(intel_dp);  	if (intel_dp->is_mst) { @@ -6835,8 +6637,6 @@ static void intel_dp_set_drrs_state(struct drm_i915_private *dev_priv,  				    const struct intel_crtc_state *crtc_state,  				    int refresh_rate)  { -	struct intel_encoder *encoder; -	struct intel_digital_port *dig_port = NULL;  	struct intel_dp *intel_dp = dev_priv->drrs.dp;  	struct intel_crtc *intel_crtc = to_intel_crtc(crtc_state->base.crtc);  	enum drrs_refresh_rate_type index = DRRS_HIGH_RR; @@ -6851,9 +6651,6 @@ static void intel_dp_set_drrs_state(struct drm_i915_private *dev_priv,  		return;  	} -	dig_port = dp_to_dig_port(intel_dp); -	encoder = &dig_port->base; -  	if (!intel_crtc) {  		DRM_DEBUG_KMS("DRRS: intel_crtc not initialized\n");  		return; @@ -7333,6 +7130,7 @@ intel_dp_init_connector(struct intel_digital_port *intel_dig_port,  	struct drm_device *dev = intel_encoder->base.dev;  	struct drm_i915_private *dev_priv = to_i915(dev);  	enum port port = intel_encoder->port; +	enum phy phy = intel_port_to_phy(dev_priv, port);  	int type;  	/* Initialize the work for modeset in case of link train failure */ @@ -7359,7 +7157,7 @@ intel_dp_init_connector(struct intel_digital_port *intel_dig_port,  		 * Currently we don't support eDP on TypeC ports, although in  		 * theory it could work on TypeC legacy ports.  		 */ -		WARN_ON(intel_port_is_tc(dev_priv, port)); +		WARN_ON(intel_phy_is_tc(dev_priv, phy));  		type = DRM_MODE_CONNECTOR_eDP;  	} else {  		type = DRM_MODE_CONNECTOR_DisplayPort; | 
