drm/xe: Wire up device shutdown handler
[ Upstream commit 501d799a47e2b83b4e41d5306c2266ea5c100a08 ] 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 Stable-dep-of: 16c1241b0875 ("drm/xe/bmg: Update Wa_16023588340") Signed-off-by: Sasha Levin <sashal@kernel.org>
This commit is contained in:
committed by
Greg Kroah-Hartman
parent
0f8df5d6f2
commit
abf32d8fa1
@@ -352,6 +352,36 @@ void xe_display_pm_suspend(struct xe_device *xe)
|
|||||||
__xe_display_pm_suspend(xe, false);
|
__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)
|
void xe_display_pm_runtime_suspend(struct xe_device *xe)
|
||||||
{
|
{
|
||||||
if (!xe->info.probe_display)
|
if (!xe->info.probe_display)
|
||||||
@@ -376,6 +406,19 @@ void xe_display_pm_suspend_late(struct xe_device *xe)
|
|||||||
intel_display_power_suspend_late(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)
|
void xe_display_pm_resume_early(struct xe_device *xe)
|
||||||
{
|
{
|
||||||
if (!xe->info.probe_display)
|
if (!xe->info.probe_display)
|
||||||
|
|||||||
@@ -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_irq_postinstall(struct xe_device *xe, struct xe_gt *gt);
|
||||||
|
|
||||||
void xe_display_pm_suspend(struct xe_device *xe);
|
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_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_early(struct xe_device *xe);
|
||||||
void xe_display_pm_resume(struct xe_device *xe);
|
void xe_display_pm_resume(struct xe_device *xe);
|
||||||
void xe_display_pm_runtime_suspend(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_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_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_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_early(struct xe_device *xe) {}
|
||||||
static inline void xe_display_pm_resume(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) {}
|
static inline void xe_display_pm_runtime_suspend(struct xe_device *xe) {}
|
||||||
|
|||||||
@@ -374,6 +374,11 @@ err:
|
|||||||
return ERR_PTR(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
|
* 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
|
* from within the driver. It is different from the PCI FLR in that it doesn't
|
||||||
@@ -387,17 +392,12 @@ err:
|
|||||||
* if/when a new instance of i915 is bound to the device it will do a full
|
* if/when a new instance of i915 is bound to the device it will do a full
|
||||||
* re-init anyway.
|
* 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 */
|
const unsigned int flr_timeout = 3 * MICRO; /* specs recommend a 3s wait */
|
||||||
struct xe_gt *gt = xe_root_mmio_gt(xe);
|
struct xe_gt *gt = xe_root_mmio_gt(xe);
|
||||||
int ret;
|
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");
|
drm_dbg(&xe->drm, "Triggering Driver-FLR\n");
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@@ -438,6 +438,16 @@ static void xe_driver_flr(struct xe_device *xe)
|
|||||||
xe_mmio_write32(gt, GU_DEBUG, DRIVERFLR_STATUS);
|
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)
|
static void xe_driver_flr_fini(void *arg)
|
||||||
{
|
{
|
||||||
struct xe_device *xe = arg;
|
struct xe_device *xe = arg;
|
||||||
@@ -797,6 +807,24 @@ void xe_device_remove(struct xe_device *xe)
|
|||||||
|
|
||||||
void xe_device_shutdown(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);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -865,6 +865,13 @@ err_msg:
|
|||||||
return err;
|
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.
|
* xe_gt_sanitize_freq() - Restore saved frequencies if necessary.
|
||||||
* @gt: the GT object
|
* @gt: the GT object
|
||||||
|
|||||||
@@ -48,6 +48,7 @@ void xe_gt_record_user_engines(struct xe_gt *gt);
|
|||||||
|
|
||||||
void xe_gt_suspend_prepare(struct xe_gt *gt);
|
void xe_gt_suspend_prepare(struct xe_gt *gt);
|
||||||
int xe_gt_suspend(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);
|
int xe_gt_resume(struct xe_gt *gt);
|
||||||
void xe_gt_reset_async(struct xe_gt *gt);
|
void xe_gt_reset_async(struct xe_gt *gt);
|
||||||
void xe_gt_sanitize(struct xe_gt *gt);
|
void xe_gt_sanitize(struct xe_gt *gt);
|
||||||
|
|||||||
Reference in New Issue
Block a user