Commit fff7b95a authored by Alan Liu's avatar Alan Liu Committed by Alex Deucher
Browse files

drm/amd/display: Fix race condition when turning off an output alone



[Why]
When 2 threads are doing commit_tail parallelly, one thread could
commit new streams to dc state but another thread remove it from dc
right away.

[How]
If we don't have new dm state change from commit_check, then we should
not call dc_commit_streams() in commit_tail. A new function
amdgpu_dm_commit_streams() is introduced to refator dc_commit_stream()
adjacent code and fix this issue.

Reviewed-by: default avatarHarry Wentland <harry.wentland@amd.com>
Acked-by: default avatarAlan Liu <haoping.liu@amd.com>
Signed-off-by: default avatarAlan Liu <haoping.liu@amd.com>
Tested-by: default avatarDaniel Wheeler <daniel.wheeler@amd.com>
Signed-off-by: default avatarAlex Deucher <alexander.deucher@amd.com>
parent c4ba2b50
Loading
Loading
Loading
Loading
+55 −59
Original line number Original line Diff line number Diff line
@@ -7927,7 +7927,6 @@ static inline uint32_t get_mem_type(struct drm_framebuffer *fb)
}
}


static void amdgpu_dm_commit_planes(struct drm_atomic_state *state,
static void amdgpu_dm_commit_planes(struct drm_atomic_state *state,
				    struct dc_state *dc_state,
				    struct drm_device *dev,
				    struct drm_device *dev,
				    struct amdgpu_display_manager *dm,
				    struct amdgpu_display_manager *dm,
				    struct drm_crtc *pcrtc,
				    struct drm_crtc *pcrtc,
@@ -8407,52 +8406,17 @@ static void amdgpu_dm_crtc_copy_transient_flags(struct drm_crtc_state *crtc_stat
	stream_state->mode_changed = drm_atomic_crtc_needs_modeset(crtc_state);
	stream_state->mode_changed = drm_atomic_crtc_needs_modeset(crtc_state);
}
}


/**
static void amdgpu_dm_commit_streams(struct drm_atomic_state *state,
 * amdgpu_dm_atomic_commit_tail() - AMDgpu DM's commit tail implementation.
					struct dc_state *dc_state)
 * @state: The atomic state to commit
 *
 * This will tell DC to commit the constructed DC state from atomic_check,
 * programming the hardware. Any failures here implies a hardware failure, since
 * atomic check should have filtered anything non-kosher.
 */
static void amdgpu_dm_atomic_commit_tail(struct drm_atomic_state *state)
{
{
	struct drm_device *dev = state->dev;
	struct drm_device *dev = state->dev;
	struct amdgpu_device *adev = drm_to_adev(dev);
	struct amdgpu_device *adev = drm_to_adev(dev);
	struct amdgpu_display_manager *dm = &adev->dm;
	struct amdgpu_display_manager *dm = &adev->dm;
	struct dm_atomic_state *dm_state;
	struct dc_state *dc_state = NULL, *dc_state_temp = NULL;
	u32 i, j;
	struct drm_crtc *crtc;
	struct drm_crtc *crtc;
	struct drm_crtc_state *old_crtc_state, *new_crtc_state;
	struct drm_crtc_state *old_crtc_state, *new_crtc_state;
	unsigned long flags;
	bool wait_for_vblank = true;
	struct drm_connector *connector;
	struct drm_connector_state *old_con_state, *new_con_state;
	struct dm_crtc_state *dm_old_crtc_state, *dm_new_crtc_state;
	struct dm_crtc_state *dm_old_crtc_state, *dm_new_crtc_state;
	int crtc_disable_count = 0;
	bool mode_set_reset_required = false;
	bool mode_set_reset_required = false;
	int r;
	u32 i;

	trace_amdgpu_dm_atomic_commit_tail_begin(state);

	r = drm_atomic_helper_wait_for_fences(dev, state, false);
	if (unlikely(r))
		DRM_ERROR("Waiting for fences timed out!");

	drm_atomic_helper_update_legacy_modeset_state(dev, state);
	drm_dp_mst_atomic_wait_for_dependencies(state);

	dm_state = dm_atomic_get_new_state(state);
	if (dm_state && dm_state->context) {
		dc_state = dm_state->context;
	} else {
		/* No state changes, retain current state. */
		dc_state_temp = dc_create_state(dm->dc);
		ASSERT(dc_state_temp);
		dc_state = dc_state_temp;
		dc_resource_state_copy_construct_current(dm->dc, dc_state);
	}


	for_each_oldnew_crtc_in_state(state, crtc, old_crtc_state,
	for_each_oldnew_crtc_in_state(state, crtc, old_crtc_state,
				      new_crtc_state, i) {
				      new_crtc_state, i) {
@@ -8551,7 +8515,6 @@ static void amdgpu_dm_atomic_commit_tail(struct drm_atomic_state *state)
		}
		}
	} /* for_each_crtc_in_state() */
	} /* for_each_crtc_in_state() */


	if (dc_state) {
	/* if there mode set or reset, disable eDP PSR */
	/* if there mode set or reset, disable eDP PSR */
	if (mode_set_reset_required) {
	if (mode_set_reset_required) {
		if (dm->vblank_control_workqueue)
		if (dm->vblank_control_workqueue)
@@ -8568,7 +8531,6 @@ static void amdgpu_dm_atomic_commit_tail(struct drm_atomic_state *state)
	if (dm->active_vblank_irq_count == 0)
	if (dm->active_vblank_irq_count == 0)
		dc_allow_idle_optimizations(dm->dc, true);
		dc_allow_idle_optimizations(dm->dc, true);
	mutex_unlock(&dm->dc_lock);
	mutex_unlock(&dm->dc_lock);
	}


	for_each_new_crtc_in_state(state, crtc, new_crtc_state, i) {
	for_each_new_crtc_in_state(state, crtc, new_crtc_state, i) {
		struct amdgpu_crtc *acrtc = to_amdgpu_crtc(crtc);
		struct amdgpu_crtc *acrtc = to_amdgpu_crtc(crtc);
@@ -8588,6 +8550,44 @@ static void amdgpu_dm_atomic_commit_tail(struct drm_atomic_state *state)
				acrtc->otg_inst = status->primary_otg_inst;
				acrtc->otg_inst = status->primary_otg_inst;
		}
		}
	}
	}
}

/**
 * amdgpu_dm_atomic_commit_tail() - AMDgpu DM's commit tail implementation.
 * @state: The atomic state to commit
 *
 * This will tell DC to commit the constructed DC state from atomic_check,
 * programming the hardware. Any failures here implies a hardware failure, since
 * atomic check should have filtered anything non-kosher.
 */
static void amdgpu_dm_atomic_commit_tail(struct drm_atomic_state *state)
{
	struct drm_device *dev = state->dev;
	struct amdgpu_device *adev = drm_to_adev(dev);
	struct amdgpu_display_manager *dm = &adev->dm;
	struct dm_atomic_state *dm_state;
	struct dc_state *dc_state = NULL;
	u32 i, j;
	struct drm_crtc *crtc;
	struct drm_crtc_state *old_crtc_state, *new_crtc_state;
	unsigned long flags;
	bool wait_for_vblank = true;
	struct drm_connector *connector;
	struct drm_connector_state *old_con_state, *new_con_state;
	struct dm_crtc_state *dm_old_crtc_state, *dm_new_crtc_state;
	int crtc_disable_count = 0;

	trace_amdgpu_dm_atomic_commit_tail_begin(state);

	drm_atomic_helper_update_legacy_modeset_state(dev, state);
	drm_dp_mst_atomic_wait_for_dependencies(state);

	dm_state = dm_atomic_get_new_state(state);
	if (dm_state && dm_state->context) {
		dc_state = dm_state->context;
		amdgpu_dm_commit_streams(state, dc_state);
	}

	for_each_oldnew_connector_in_state(state, connector, old_con_state, new_con_state, i) {
	for_each_oldnew_connector_in_state(state, connector, old_con_state, new_con_state, i) {
		struct dm_connector_state *dm_new_con_state = to_dm_connector_state(new_con_state);
		struct dm_connector_state *dm_new_con_state = to_dm_connector_state(new_con_state);
		struct amdgpu_crtc *acrtc = to_amdgpu_crtc(dm_new_con_state->base.crtc);
		struct amdgpu_crtc *acrtc = to_amdgpu_crtc(dm_new_con_state->base.crtc);
@@ -8865,8 +8865,7 @@ static void amdgpu_dm_atomic_commit_tail(struct drm_atomic_state *state)
		dm_new_crtc_state = to_dm_crtc_state(new_crtc_state);
		dm_new_crtc_state = to_dm_crtc_state(new_crtc_state);


		if (dm_new_crtc_state->stream)
		if (dm_new_crtc_state->stream)
			amdgpu_dm_commit_planes(state, dc_state, dev,
			amdgpu_dm_commit_planes(state, dev, dm, crtc, wait_for_vblank);
						dm, crtc, wait_for_vblank);
	}
	}


	/* Update audio instances for each connector. */
	/* Update audio instances for each connector. */
@@ -8921,9 +8920,6 @@ static void amdgpu_dm_atomic_commit_tail(struct drm_atomic_state *state)
	for (i = 0; i < crtc_disable_count; i++)
	for (i = 0; i < crtc_disable_count; i++)
		pm_runtime_put_autosuspend(dev->dev);
		pm_runtime_put_autosuspend(dev->dev);
	pm_runtime_mark_last_busy(dev->dev);
	pm_runtime_mark_last_busy(dev->dev);

	if (dc_state_temp)
		dc_release_state(dc_state_temp);
}
}


static int dm_force_atomic_commit(struct drm_connector *connector)
static int dm_force_atomic_commit(struct drm_connector *connector)