159c6106eSJani Nikula // SPDX-License-Identifier: MIT
259c6106eSJani Nikula /*
359c6106eSJani Nikula  * Copyright © 2023 Intel Corporation
459c6106eSJani Nikula  */
559c6106eSJani Nikula 
659c6106eSJani Nikula #include <drm/drm_atomic_helper.h>
759c6106eSJani Nikula 
859c6106eSJani Nikula #include "i915_drv.h"
959c6106eSJani Nikula #include "intel_clock_gating.h"
1059c6106eSJani Nikula #include "intel_display_driver.h"
1159c6106eSJani Nikula #include "intel_display_reset.h"
1259c6106eSJani Nikula #include "intel_display_types.h"
1359c6106eSJani Nikula #include "intel_hotplug.h"
1459c6106eSJani Nikula #include "intel_pps.h"
1559c6106eSJani Nikula 
gpu_reset_clobbers_display(struct drm_i915_private * dev_priv)1659c6106eSJani Nikula static bool gpu_reset_clobbers_display(struct drm_i915_private *dev_priv)
1759c6106eSJani Nikula {
1859c6106eSJani Nikula 	return (INTEL_INFO(dev_priv)->gpu_reset_clobbers_display &&
1959c6106eSJani Nikula 		intel_has_gpu_reset(to_gt(dev_priv)));
2059c6106eSJani Nikula }
2159c6106eSJani Nikula 
intel_display_reset_prepare(struct drm_i915_private * dev_priv)2259c6106eSJani Nikula void intel_display_reset_prepare(struct drm_i915_private *dev_priv)
2359c6106eSJani Nikula {
2459c6106eSJani Nikula 	struct drm_modeset_acquire_ctx *ctx = &dev_priv->display.restore.reset_ctx;
2559c6106eSJani Nikula 	struct drm_atomic_state *state;
2659c6106eSJani Nikula 	int ret;
2759c6106eSJani Nikula 
2859c6106eSJani Nikula 	if (!HAS_DISPLAY(dev_priv))
2959c6106eSJani Nikula 		return;
3059c6106eSJani Nikula 
3159c6106eSJani Nikula 	/* reset doesn't touch the display */
3259c6106eSJani Nikula 	if (!dev_priv->params.force_reset_modeset_test &&
3359c6106eSJani Nikula 	    !gpu_reset_clobbers_display(dev_priv))
3459c6106eSJani Nikula 		return;
3559c6106eSJani Nikula 
3659c6106eSJani Nikula 	/* We have a modeset vs reset deadlock, defensively unbreak it. */
3759c6106eSJani Nikula 	set_bit(I915_RESET_MODESET, &to_gt(dev_priv)->reset.flags);
3859c6106eSJani Nikula 	smp_mb__after_atomic();
3959c6106eSJani Nikula 	wake_up_bit(&to_gt(dev_priv)->reset.flags, I915_RESET_MODESET);
4059c6106eSJani Nikula 
4159c6106eSJani Nikula 	if (atomic_read(&dev_priv->gpu_error.pending_fb_pin)) {
4259c6106eSJani Nikula 		drm_dbg_kms(&dev_priv->drm,
4359c6106eSJani Nikula 			    "Modeset potentially stuck, unbreaking through wedging\n");
4459c6106eSJani Nikula 		intel_gt_set_wedged(to_gt(dev_priv));
4559c6106eSJani Nikula 	}
4659c6106eSJani Nikula 
4759c6106eSJani Nikula 	/*
4859c6106eSJani Nikula 	 * Need mode_config.mutex so that we don't
4959c6106eSJani Nikula 	 * trample ongoing ->detect() and whatnot.
5059c6106eSJani Nikula 	 */
5159c6106eSJani Nikula 	mutex_lock(&dev_priv->drm.mode_config.mutex);
5259c6106eSJani Nikula 	drm_modeset_acquire_init(ctx, 0);
5359c6106eSJani Nikula 	while (1) {
5459c6106eSJani Nikula 		ret = drm_modeset_lock_all_ctx(&dev_priv->drm, ctx);
5559c6106eSJani Nikula 		if (ret != -EDEADLK)
5659c6106eSJani Nikula 			break;
5759c6106eSJani Nikula 
5859c6106eSJani Nikula 		drm_modeset_backoff(ctx);
5959c6106eSJani Nikula 	}
6059c6106eSJani Nikula 	/*
6159c6106eSJani Nikula 	 * Disabling the crtcs gracefully seems nicer. Also the
6259c6106eSJani Nikula 	 * g33 docs say we should at least disable all the planes.
6359c6106eSJani Nikula 	 */
6459c6106eSJani Nikula 	state = drm_atomic_helper_duplicate_state(&dev_priv->drm, ctx);
6559c6106eSJani Nikula 	if (IS_ERR(state)) {
6659c6106eSJani Nikula 		ret = PTR_ERR(state);
6759c6106eSJani Nikula 		drm_err(&dev_priv->drm, "Duplicating state failed with %i\n",
6859c6106eSJani Nikula 			ret);
6959c6106eSJani Nikula 		return;
7059c6106eSJani Nikula 	}
7159c6106eSJani Nikula 
7259c6106eSJani Nikula 	ret = drm_atomic_helper_disable_all(&dev_priv->drm, ctx);
7359c6106eSJani Nikula 	if (ret) {
7459c6106eSJani Nikula 		drm_err(&dev_priv->drm, "Suspending crtc's failed with %i\n",
7559c6106eSJani Nikula 			ret);
7659c6106eSJani Nikula 		drm_atomic_state_put(state);
7759c6106eSJani Nikula 		return;
7859c6106eSJani Nikula 	}
7959c6106eSJani Nikula 
8059c6106eSJani Nikula 	dev_priv->display.restore.modeset_state = state;
8159c6106eSJani Nikula 	state->acquire_ctx = ctx;
8259c6106eSJani Nikula }
8359c6106eSJani Nikula 
intel_display_reset_finish(struct drm_i915_private * i915)8459c6106eSJani Nikula void intel_display_reset_finish(struct drm_i915_private *i915)
8559c6106eSJani Nikula {
8659c6106eSJani Nikula 	struct drm_modeset_acquire_ctx *ctx = &i915->display.restore.reset_ctx;
8759c6106eSJani Nikula 	struct drm_atomic_state *state;
8859c6106eSJani Nikula 	int ret;
8959c6106eSJani Nikula 
9059c6106eSJani Nikula 	if (!HAS_DISPLAY(i915))
9159c6106eSJani Nikula 		return;
9259c6106eSJani Nikula 
9359c6106eSJani Nikula 	/* reset doesn't touch the display */
9459c6106eSJani Nikula 	if (!test_bit(I915_RESET_MODESET, &to_gt(i915)->reset.flags))
9559c6106eSJani Nikula 		return;
9659c6106eSJani Nikula 
9759c6106eSJani Nikula 	state = fetch_and_zero(&i915->display.restore.modeset_state);
9859c6106eSJani Nikula 	if (!state)
9959c6106eSJani Nikula 		goto unlock;
10059c6106eSJani Nikula 
10159c6106eSJani Nikula 	/* reset doesn't touch the display */
10259c6106eSJani Nikula 	if (!gpu_reset_clobbers_display(i915)) {
10359c6106eSJani Nikula 		/* for testing only restore the display */
10459c6106eSJani Nikula 		ret = drm_atomic_helper_commit_duplicated_state(state, ctx);
10559c6106eSJani Nikula 		if (ret) {
10659c6106eSJani Nikula 			drm_WARN_ON(&i915->drm, ret == -EDEADLK);
10759c6106eSJani Nikula 			drm_err(&i915->drm,
10859c6106eSJani Nikula 				"Restoring old state failed with %i\n", ret);
10959c6106eSJani Nikula 		}
11059c6106eSJani Nikula 	} else {
11159c6106eSJani Nikula 		/*
11259c6106eSJani Nikula 		 * The display has been reset as well,
11359c6106eSJani Nikula 		 * so need a full re-initialization.
11459c6106eSJani Nikula 		 */
11559c6106eSJani Nikula 		intel_pps_unlock_regs_wa(i915);
11659c6106eSJani Nikula 		intel_display_driver_init_hw(i915);
11759c6106eSJani Nikula 		intel_clock_gating_init(i915);
11859c6106eSJani Nikula 		intel_hpd_init(i915);
11959c6106eSJani Nikula 
120*cde4bd87SJani Nikula 		ret = __intel_display_driver_resume(i915, state, ctx);
12159c6106eSJani Nikula 		if (ret)
12259c6106eSJani Nikula 			drm_err(&i915->drm,
12359c6106eSJani Nikula 				"Restoring old state failed with %i\n", ret);
12459c6106eSJani Nikula 
12559c6106eSJani Nikula 		intel_hpd_poll_disable(i915);
12659c6106eSJani Nikula 	}
12759c6106eSJani Nikula 
12859c6106eSJani Nikula 	drm_atomic_state_put(state);
12959c6106eSJani Nikula unlock:
13059c6106eSJani Nikula 	drm_modeset_drop_locks(ctx);
13159c6106eSJani Nikula 	drm_modeset_acquire_fini(ctx);
13259c6106eSJani Nikula 	mutex_unlock(&i915->drm.mode_config.mutex);
13359c6106eSJani Nikula 
13459c6106eSJani Nikula 	clear_bit_unlock(I915_RESET_MODESET, &to_gt(i915)->reset.flags);
13559c6106eSJani Nikula }
136