The ->detect() vfunc of connectors is usually responsible for setting the encoder type on intel_digital_ports when a hotplug event happens. However we sometimes want to force a modeset on a specific connector: - the user can ask the SETCRTC ioctl to use a connector that isn't marked as connected (because we never received a hotplug event for it). This can be also used in tests to do a modeset without the need of a plugged-in monitor. - the command line video= option can be used to force modesets, eg.: video=HDMI-A-1:1024x768e So, before we try to do anything with the DDI encoder as part of a modeset, we need to ensure that the personality of the encoder matches the selected connector. v2: Made by Paulo: - Return false if we have more than one connector - WARN in case it's not eDP/DP/HDMI - Don't break error strings in pieces - Change the "status == connected" message from DRM_ERROR to DRM_DEBUG_KMS - Don't store+use the old encoder->type at compute_config - Add bugzilla reference - Add testcase reference v3: Made by Paulo: - WARN in case "connectors > 1", since we have a patch that should catch this case earlier v4: Authorize unknown encoders with a "connected" connector to change personality (Damien) Testcase: igt/kms_setmode/clone-exclusive-crtc Bugzilla: https://bugs.freedesktop.org/show_bug.cgi?id=68463 Tested-by: Paulo Zanoni <paulo.r.zanoni@xxxxxxxxx> Signed-off-by: Damien Lespiau <damien.lespiau@xxxxxxxxx> Signed-off-by: Paulo Zanoni <paulo.r.zanoni@xxxxxxxxx> --- drivers/gpu/drm/i915/intel_ddi.c | 102 +++++++++++++++++++++++++++++++++++++-- 1 file changed, 99 insertions(+), 3 deletions(-) diff --git a/drivers/gpu/drm/i915/intel_ddi.c b/drivers/gpu/drm/i915/intel_ddi.c index 74749c6..79776f8 100644 --- a/drivers/gpu/drm/i915/intel_ddi.c +++ b/drivers/gpu/drm/i915/intel_ddi.c @@ -1518,18 +1518,114 @@ static void intel_ddi_destroy(struct drm_encoder *encoder) intel_dp_encoder_destroy(encoder); } +/* + * The ->detect() vfunc of connectors is usually responsible for setting the + * encoder type on intel_digital_ports when a hotplug event happens. + * + * However we sometimes want to force a modeset on a specific connector: + * - the user can ask the SETCRTC ioctl to use a connector that isn't marked + * as connected (because we never received a hotplug event for it). + * This can be also used in tests to do a modeset without the need of a + * plugged-in monitor. + * - the command line video= option can be used to force modesets, eg.: + * video=HDMI-A-1:1024x768e + * + * So, before we try to do anything with the DDI encoder as part of a modeset, + * we need to ensure that the personality of the encoder matches the selected + * connector. + */ +static bool +intel_ddi_ensure_encoder_type(struct intel_encoder *encoder) +{ + struct drm_device *dev = encoder->base.dev; + struct intel_connector *connector; + int connectors = 0; + + list_for_each_entry(connector, &dev->mode_config.connector_list, + base.head) { + int connector_type, old_encoder_type, new_encoder_type; + int port; + + if (connector->new_encoder != encoder) + continue; + + connectors++; + if (WARN_ON(connectors > 1)) + return false; + + connector_type = connector->base.connector_type; + old_encoder_type = encoder->type; + switch (connector_type) { + case DRM_MODE_CONNECTOR_HDMIA: + case DRM_MODE_CONNECTOR_HDMIB: + new_encoder_type = INTEL_OUTPUT_HDMI; + break; + case DRM_MODE_CONNECTOR_DisplayPort: + new_encoder_type = INTEL_OUTPUT_DISPLAYPORT; + break; + case DRM_MODE_CONNECTOR_eDP: + continue; + default: + WARN(1, "DRM connector type %d\n", connector_type); + continue; + } + + if (old_encoder_type == new_encoder_type) + continue; + + port = intel_ddi_get_encoder_port(encoder); + + if (old_encoder_type == INTEL_OUTPUT_EDP) { + DRM_ERROR("Can't change DDI %c personality to %s, it's an eDP DDI\n", + port_name(port), + intel_output_name(new_encoder_type)); + return false; + } + + /* + * We can't change the DDI type if we already have a connected + * device on this port. The first time a DDI is used though + * (encoder_type is INTEL_OUTPUT_UNKNOWN) and we force a + * connector to be connected (either trought the kernel command + * line or KMS) we need to comply. + */ + if (old_encoder_type != INTEL_OUTPUT_UNKNOWN && + connector->base.status == connector_status_connected) { + DRM_DEBUG_KMS("Can't change DDI %c personality to %s, it has a connected %s device\n", + port_name(port), + intel_output_name(new_encoder_type), + intel_output_name(old_encoder_type)); + return false; + } + + DRM_DEBUG_KMS("Changing DDI %c from %s to %s\n", + port_name(port), + intel_output_name(old_encoder_type), + intel_output_name(new_encoder_type)); + + encoder->type = new_encoder_type; + } + + return true; +} + static bool intel_ddi_compute_config(struct intel_encoder *encoder, struct intel_crtc_config *pipe_config) { - int type = encoder->type; int port = intel_ddi_get_encoder_port(encoder); + int ret; + + ret = intel_ddi_ensure_encoder_type(encoder); + if (!ret) + return false; - WARN(type == INTEL_OUTPUT_UNKNOWN, "compute_config() on unknown output!\n"); + WARN(encoder->type == INTEL_OUTPUT_UNKNOWN, + "compute_config() on unknown output!\n"); if (port == PORT_A) pipe_config->cpu_transcoder = TRANSCODER_EDP; - if (type == INTEL_OUTPUT_HDMI) + if (encoder->type == INTEL_OUTPUT_HDMI) return intel_hdmi_compute_config(encoder, pipe_config); else return intel_dp_compute_config(encoder, pipe_config); -- 1.8.3.1 _______________________________________________ Intel-gfx mailing list Intel-gfx@xxxxxxxxxxxxxxxxxxxxx http://lists.freedesktop.org/mailman/listinfo/intel-gfx