]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
drm/xe: Wire up device shutdown handler
authorMaarten Lankhorst <maarten.lankhorst@linux.intel.com>
Thu, 5 Sep 2024 15:00:51 +0000 (17:00 +0200)
committerMaarten Lankhorst <maarten.lankhorst@linux.intel.com>
Wed, 11 Sep 2024 17:07:57 +0000 (19:07 +0200)
The system is turning off, and we should probably put the device
in a safe power state. We don't need to evict VRAM or suspend running
jobs to a safe state, as the device is rebooted anyway.

This does not imply the system is necessarily reset, as we can
kexec into a new kernel. Without shutting down, things like
USB Type-C may mysteriously start failing.

References: https://gitlab.freedesktop.org/drm/i915/kernel/-/issues/3500
Signed-off-by: Maarten Lankhorst <maarten.lankhorst@linux.intel.com>
[mlankhorst: Add !xe_driver_flr_disabled assert]
Reviewed-by: Rodrigo Vivi <rodrigo.vivi@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20240905150052.174895-4-maarten.lankhorst@linux.intel.com
drivers/gpu/drm/xe/display/xe_display.c
drivers/gpu/drm/xe/display/xe_display.h
drivers/gpu/drm/xe/xe_device.c
drivers/gpu/drm/xe/xe_gt.c
drivers/gpu/drm/xe/xe_gt.h

index 1c25c4f6a53b3882d6b2871c3956b9b4e8166232..86009a17e4c25d6fe59f5df5fe8c5e69dcc22074 100644 (file)
@@ -349,6 +349,36 @@ void xe_display_pm_suspend(struct xe_device *xe)
        __xe_display_pm_suspend(xe, false);
 }
 
+void xe_display_pm_shutdown(struct xe_device *xe)
+{
+       struct intel_display *display = &xe->display;
+
+       if (!xe->info.probe_display)
+               return;
+
+       intel_power_domains_disable(xe);
+       intel_fbdev_set_suspend(&xe->drm, FBINFO_STATE_SUSPENDED, true);
+       if (has_display(xe)) {
+               drm_kms_helper_poll_disable(&xe->drm);
+               intel_display_driver_disable_user_access(xe);
+               intel_display_driver_suspend(xe);
+       }
+
+       xe_display_flush_cleanup_work(xe);
+       intel_dp_mst_suspend(xe);
+       intel_hpd_cancel_work(xe);
+
+       if (has_display(xe))
+               intel_display_driver_suspend_access(xe);
+
+       intel_encoder_suspend_all(display);
+       intel_encoder_shutdown_all(display);
+
+       intel_opregion_suspend(display, PCI_D3cold);
+
+       intel_dmc_suspend(xe);
+}
+
 void xe_display_pm_runtime_suspend(struct xe_device *xe)
 {
        if (!xe->info.probe_display)
@@ -371,6 +401,19 @@ void xe_display_pm_suspend_late(struct xe_device *xe)
        intel_display_power_suspend_late(xe);
 }
 
+void xe_display_pm_shutdown_late(struct xe_device *xe)
+{
+       if (!xe->info.probe_display)
+               return;
+
+       /*
+        * The only requirement is to reboot with display DC states disabled,
+        * for now leaving all display power wells in the INIT power domain
+        * enabled.
+        */
+       intel_power_domains_driver_remove(xe);
+}
+
 void xe_display_pm_resume_early(struct xe_device *xe)
 {
        if (!xe->info.probe_display)
index bed55fd26f30444517a85efb921c786edc6429d0..17afa537aee50882de91a059a18c4e93bd256084 100644 (file)
@@ -35,7 +35,9 @@ void xe_display_irq_reset(struct xe_device *xe);
 void xe_display_irq_postinstall(struct xe_device *xe, struct xe_gt *gt);
 
 void xe_display_pm_suspend(struct xe_device *xe);
+void xe_display_pm_shutdown(struct xe_device *xe);
 void xe_display_pm_suspend_late(struct xe_device *xe);
+void xe_display_pm_shutdown_late(struct xe_device *xe);
 void xe_display_pm_resume_early(struct xe_device *xe);
 void xe_display_pm_resume(struct xe_device *xe);
 void xe_display_pm_runtime_suspend(struct xe_device *xe);
@@ -66,7 +68,9 @@ static inline void xe_display_irq_reset(struct xe_device *xe) {}
 static inline void xe_display_irq_postinstall(struct xe_device *xe, struct xe_gt *gt) {}
 
 static inline void xe_display_pm_suspend(struct xe_device *xe) {}
+static inline void xe_display_pm_shutdown(struct xe_device *xe) {}
 static inline void xe_display_pm_suspend_late(struct xe_device *xe) {}
+static inline void xe_display_pm_shutdown_late(struct xe_device *xe) {}
 static inline void xe_display_pm_resume_early(struct xe_device *xe) {}
 static inline void xe_display_pm_resume(struct xe_device *xe) {}
 static inline void xe_display_pm_runtime_suspend(struct xe_device *xe) {}
index 449b85035d3a26f1c6366996185bd48753db465e..fdb3a4133f43d83609e9058c8de36105d5601a1a 100644 (file)
@@ -383,6 +383,11 @@ err:
        return ERR_PTR(err);
 }
 
+static bool xe_driver_flr_disabled(struct xe_device *xe)
+{
+       return xe_mmio_read32(xe_root_mmio_gt(xe), GU_CNTL_PROTECTED) & DRIVERINT_FLR_DIS;
+}
+
 /*
  * The driver-initiated FLR is the highest level of reset that we can trigger
  * from within the driver. It is different from the PCI FLR in that it doesn't
@@ -396,17 +401,12 @@ err:
  * if/when a new instance of i915 is bound to the device it will do a full
  * re-init anyway.
  */
-static void xe_driver_flr(struct xe_device *xe)
+static void __xe_driver_flr(struct xe_device *xe)
 {
        const unsigned int flr_timeout = 3 * MICRO; /* specs recommend a 3s wait */
        struct xe_gt *gt = xe_root_mmio_gt(xe);
        int ret;
 
-       if (xe_mmio_read32(gt, GU_CNTL_PROTECTED) & DRIVERINT_FLR_DIS) {
-               drm_info_once(&xe->drm, "BIOS Disabled Driver-FLR\n");
-               return;
-       }
-
        drm_dbg(&xe->drm, "Triggering Driver-FLR\n");
 
        /*
@@ -447,6 +447,16 @@ static void xe_driver_flr(struct xe_device *xe)
        xe_mmio_write32(gt, GU_DEBUG, DRIVERFLR_STATUS);
 }
 
+static void xe_driver_flr(struct xe_device *xe)
+{
+       if (xe_driver_flr_disabled(xe)) {
+               drm_info_once(&xe->drm, "BIOS Disabled Driver-FLR\n");
+               return;
+       }
+
+       __xe_driver_flr(xe);
+}
+
 static void xe_driver_flr_fini(void *arg)
 {
        struct xe_device *xe = arg;
@@ -800,6 +810,24 @@ void xe_device_remove(struct xe_device *xe)
 
 void xe_device_shutdown(struct xe_device *xe)
 {
+       struct xe_gt *gt;
+       u8 id;
+
+       drm_dbg(&xe->drm, "Shutting down device\n");
+
+       if (xe_driver_flr_disabled(xe)) {
+               xe_display_pm_shutdown(xe);
+
+               xe_irq_suspend(xe);
+
+               for_each_gt(gt, xe, id)
+                       xe_gt_shutdown(gt);
+
+               xe_display_pm_shutdown_late(xe);
+       } else {
+               /* BOOM! */
+               __xe_driver_flr(xe);
+       }
 }
 
 /**
index f0dc2bf24c7b172057e83e86c8b8aae718cb5014..9e338f4e5fb292c3320ba8b8573e18ebe4ac55c9 100644 (file)
@@ -862,6 +862,13 @@ err_msg:
        return err;
 }
 
+void xe_gt_shutdown(struct xe_gt *gt)
+{
+       xe_force_wake_get(gt_to_fw(gt), XE_FORCEWAKE_ALL);
+       do_gt_reset(gt);
+       xe_force_wake_put(gt_to_fw(gt), XE_FORCEWAKE_ALL);
+}
+
 /**
  * xe_gt_sanitize_freq() - Restore saved frequencies if necessary.
  * @gt: the GT object
index 8b1a5027dcf271953e96e97e6fe6657535b6b8b7..97def44afa4c86ad259263c98d463ab759378d26 100644 (file)
@@ -54,6 +54,7 @@ void xe_gt_record_user_engines(struct xe_gt *gt);
 
 void xe_gt_suspend_prepare(struct xe_gt *gt);
 int xe_gt_suspend(struct xe_gt *gt);
+void xe_gt_shutdown(struct xe_gt *gt);
 int xe_gt_resume(struct xe_gt *gt);
 void xe_gt_reset_async(struct xe_gt *gt);
 void xe_gt_sanitize(struct xe_gt *gt);