mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	drm/nouveau/kms: Don't change EDID when it hasn't actually changed
Currently in nouveau_connector_ddc_detect() and
nouveau_connector_detect_lvds(), we start the connector probing process
by releasing the previous EDID and informing DRM of the change. However,
since commit 5186421cbf ("drm: Introduce epoch counter to
drm_connector") drm_connector_update_edid_property() actually checks
whether the new EDID we've specified is different from the previous one,
and updates the connector's epoch accordingly if it is. But, because we
always set the EDID to NULL first in nouveau_connector_ddc_detect() and
nouveau_connector_detect_lvds() we end up making DRM think that the EDID
changes every single time we do a connector probe - which isn't needed.
So, let's fix this by not clearing the EDID at the start of the
connector probing process, and instead simply changing or removing it
once near the end of the probing process. This will help prevent us from
sending unneeded hotplug events to userspace when nothing has actually
changed.
Signed-off-by: Lyude Paul <lyude@redhat.com>
Reviewed-by: Ben Skeggs <bskeggs@redhat.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20200826182456.322681-19-lyude@redhat.com
			
			
This commit is contained in:
		
							parent
							
								
									a4efad354c
								
							
						
					
					
						commit
						f28e32d390
					
				| @ -528,6 +528,17 @@ nouveau_connector_set_encoder(struct drm_connector *connector, | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static void | ||||||
|  | nouveau_connector_set_edid(struct nouveau_connector *nv_connector, | ||||||
|  | 			   struct edid *edid) | ||||||
|  | { | ||||||
|  | 	struct edid *old_edid = nv_connector->edid; | ||||||
|  | 
 | ||||||
|  | 	drm_connector_update_edid_property(&nv_connector->base, edid); | ||||||
|  | 	kfree(old_edid); | ||||||
|  | 	nv_connector->edid = edid; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| static enum drm_connector_status | static enum drm_connector_status | ||||||
| nouveau_connector_detect(struct drm_connector *connector, bool force) | nouveau_connector_detect(struct drm_connector *connector, bool force) | ||||||
| { | { | ||||||
| @ -541,13 +552,6 @@ nouveau_connector_detect(struct drm_connector *connector, bool force) | |||||||
| 	int ret; | 	int ret; | ||||||
| 	enum drm_connector_status conn_status = connector_status_disconnected; | 	enum drm_connector_status conn_status = connector_status_disconnected; | ||||||
| 
 | 
 | ||||||
| 	/* Cleanup the previous EDID block. */ |  | ||||||
| 	if (nv_connector->edid) { |  | ||||||
| 		drm_connector_update_edid_property(connector, NULL); |  | ||||||
| 		kfree(nv_connector->edid); |  | ||||||
| 		nv_connector->edid = NULL; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	/* Outputs are only polled while runtime active, so resuming the
 | 	/* Outputs are only polled while runtime active, so resuming the
 | ||||||
| 	 * device here is unnecessary (and would deadlock upon runtime suspend | 	 * device here is unnecessary (and would deadlock upon runtime suspend | ||||||
| 	 * because it waits for polling to finish). We do however, want to | 	 * because it waits for polling to finish). We do however, want to | ||||||
| @ -560,22 +564,23 @@ nouveau_connector_detect(struct drm_connector *connector, bool force) | |||||||
| 		ret = pm_runtime_get_sync(dev->dev); | 		ret = pm_runtime_get_sync(dev->dev); | ||||||
| 		if (ret < 0 && ret != -EACCES) { | 		if (ret < 0 && ret != -EACCES) { | ||||||
| 			pm_runtime_put_autosuspend(dev->dev); | 			pm_runtime_put_autosuspend(dev->dev); | ||||||
|  | 			nouveau_connector_set_edid(nv_connector, NULL); | ||||||
| 			return conn_status; | 			return conn_status; | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	nv_encoder = nouveau_connector_ddc_detect(connector); | 	nv_encoder = nouveau_connector_ddc_detect(connector); | ||||||
| 	if (nv_encoder && (i2c = nv_encoder->i2c) != NULL) { | 	if (nv_encoder && (i2c = nv_encoder->i2c) != NULL) { | ||||||
|  | 		struct edid *new_edid; | ||||||
|  | 
 | ||||||
| 		if ((vga_switcheroo_handler_flags() & | 		if ((vga_switcheroo_handler_flags() & | ||||||
| 		     VGA_SWITCHEROO_CAN_SWITCH_DDC) && | 		     VGA_SWITCHEROO_CAN_SWITCH_DDC) && | ||||||
| 		    nv_connector->type == DCB_CONNECTOR_LVDS) | 		    nv_connector->type == DCB_CONNECTOR_LVDS) | ||||||
| 			nv_connector->edid = drm_get_edid_switcheroo(connector, | 			new_edid = drm_get_edid_switcheroo(connector, i2c); | ||||||
| 								     i2c); |  | ||||||
| 		else | 		else | ||||||
| 			nv_connector->edid = drm_get_edid(connector, i2c); | 			new_edid = drm_get_edid(connector, i2c); | ||||||
| 
 | 
 | ||||||
| 		drm_connector_update_edid_property(connector, | 		nouveau_connector_set_edid(nv_connector, new_edid); | ||||||
| 							nv_connector->edid); |  | ||||||
| 		if (!nv_connector->edid) { | 		if (!nv_connector->edid) { | ||||||
| 			NV_ERROR(drm, "DDC responded, but no EDID for %s\n", | 			NV_ERROR(drm, "DDC responded, but no EDID for %s\n", | ||||||
| 				 connector->name); | 				 connector->name); | ||||||
| @ -609,6 +614,8 @@ nouveau_connector_detect(struct drm_connector *connector, bool force) | |||||||
| 		conn_status = connector_status_connected; | 		conn_status = connector_status_connected; | ||||||
| 		drm_dp_cec_set_edid(&nv_connector->aux, nv_connector->edid); | 		drm_dp_cec_set_edid(&nv_connector->aux, nv_connector->edid); | ||||||
| 		goto out; | 		goto out; | ||||||
|  | 	} else { | ||||||
|  | 		nouveau_connector_set_edid(nv_connector, NULL); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	nv_encoder = nouveau_connector_of_detect(connector); | 	nv_encoder = nouveau_connector_of_detect(connector); | ||||||
| @ -652,18 +659,12 @@ nouveau_connector_detect_lvds(struct drm_connector *connector, bool force) | |||||||
| 	struct nouveau_drm *drm = nouveau_drm(dev); | 	struct nouveau_drm *drm = nouveau_drm(dev); | ||||||
| 	struct nouveau_connector *nv_connector = nouveau_connector(connector); | 	struct nouveau_connector *nv_connector = nouveau_connector(connector); | ||||||
| 	struct nouveau_encoder *nv_encoder = NULL; | 	struct nouveau_encoder *nv_encoder = NULL; | ||||||
|  | 	struct edid *edid = NULL; | ||||||
| 	enum drm_connector_status status = connector_status_disconnected; | 	enum drm_connector_status status = connector_status_disconnected; | ||||||
| 
 | 
 | ||||||
| 	/* Cleanup the previous EDID block. */ |  | ||||||
| 	if (nv_connector->edid) { |  | ||||||
| 		drm_connector_update_edid_property(connector, NULL); |  | ||||||
| 		kfree(nv_connector->edid); |  | ||||||
| 		nv_connector->edid = NULL; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	nv_encoder = find_encoder(connector, DCB_OUTPUT_LVDS); | 	nv_encoder = find_encoder(connector, DCB_OUTPUT_LVDS); | ||||||
| 	if (!nv_encoder) | 	if (!nv_encoder) | ||||||
| 		return connector_status_disconnected; | 		goto out; | ||||||
| 
 | 
 | ||||||
| 	/* Try retrieving EDID via DDC */ | 	/* Try retrieving EDID via DDC */ | ||||||
| 	if (!drm->vbios.fp_no_ddc) { | 	if (!drm->vbios.fp_no_ddc) { | ||||||
| @ -682,7 +683,8 @@ nouveau_connector_detect_lvds(struct drm_connector *connector, bool force) | |||||||
| 	 * valid - it's not (rh#613284) | 	 * valid - it's not (rh#613284) | ||||||
| 	 */ | 	 */ | ||||||
| 	if (nv_encoder->dcb->lvdsconf.use_acpi_for_edid) { | 	if (nv_encoder->dcb->lvdsconf.use_acpi_for_edid) { | ||||||
| 		if ((nv_connector->edid = nouveau_acpi_edid(dev, connector))) { | 		edid = nouveau_acpi_edid(dev, connector); | ||||||
|  | 		if (edid) { | ||||||
| 			status = connector_status_connected; | 			status = connector_status_connected; | ||||||
| 			goto out; | 			goto out; | ||||||
| 		} | 		} | ||||||
| @ -702,12 +704,10 @@ nouveau_connector_detect_lvds(struct drm_connector *connector, bool force) | |||||||
| 	 * stored for the panel stored in them. | 	 * stored for the panel stored in them. | ||||||
| 	 */ | 	 */ | ||||||
| 	if (!drm->vbios.fp_no_ddc) { | 	if (!drm->vbios.fp_no_ddc) { | ||||||
| 		struct edid *edid = | 		edid = (struct edid *)nouveau_bios_embedded_edid(dev); | ||||||
| 			(struct edid *)nouveau_bios_embedded_edid(dev); |  | ||||||
| 		if (edid) { | 		if (edid) { | ||||||
| 			nv_connector->edid = | 			edid = kmemdup(edid, EDID_LENGTH, GFP_KERNEL); | ||||||
| 					kmemdup(edid, EDID_LENGTH, GFP_KERNEL); | 			if (edid) | ||||||
| 			if (nv_connector->edid) |  | ||||||
| 				status = connector_status_connected; | 				status = connector_status_connected; | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @ -720,7 +720,7 @@ out: | |||||||
| 		status = connector_status_unknown; | 		status = connector_status_unknown; | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
| 	drm_connector_update_edid_property(connector, nv_connector->edid); | 	nouveau_connector_set_edid(nv_connector, edid); | ||||||
| 	nouveau_connector_set_encoder(connector, nv_encoder); | 	nouveau_connector_set_encoder(connector, nv_encoder); | ||||||
| 	return status; | 	return status; | ||||||
| } | } | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Lyude Paul
						Lyude Paul