Amdgpu preprocesses ras interrupts.

Signed-off-by: YiPeng Chai <yipeng.c...@amd.com>
Reviewed-by: Tao Zhou <tao.zh...@amd.com>
---
 .../drm/amd/ras/ras_mgr/amdgpu_ras_process.c  | 126 ++++++++++++++++++
 .../drm/amd/ras/ras_mgr/amdgpu_ras_process.h  |  37 +++++
 2 files changed, 163 insertions(+)
 create mode 100644 drivers/gpu/drm/amd/ras/ras_mgr/amdgpu_ras_process.c
 create mode 100644 drivers/gpu/drm/amd/ras/ras_mgr/amdgpu_ras_process.h

diff --git a/drivers/gpu/drm/amd/ras/ras_mgr/amdgpu_ras_process.c 
b/drivers/gpu/drm/amd/ras/ras_mgr/amdgpu_ras_process.c
new file mode 100644
index 000000000000..6727fc9a2b9b
--- /dev/null
+++ b/drivers/gpu/drm/amd/ras/ras_mgr/amdgpu_ras_process.c
@@ -0,0 +1,126 @@
+// SPDX-License-Identifier: MIT
+/*
+ * Copyright (c) 2025 Advanced Micro Devices, Inc. All rights reserved.
+ *
+ * Permission is hereby granted, free of charge, to any person obtaining a copy
+ * of this software and associated documentation files (the "Software"), to 
deal
+ * in the Software without restriction, including without limitation the rights
+ * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+ * copies of the Software, and to permit persons to whom the Software is
+ * furnished to do so, subject to the following conditions:
+ *
+ * The above copyright notice and this permission notice shall be included in
+ * all copies or substantial portions of the Software.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+ * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+ * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL THE
+ * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+ * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING 
FROM,
+ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
+ * THE SOFTWARE.
+ */
+
+#include "amdgpu.h"
+#include "amdgpu_reset.h"
+#include "amdgpu_xgmi.h"
+#include "ras_sys.h"
+#include "amdgpu_ras_mgr.h"
+#include "amdgpu_ras_process.h"
+
+#define RAS_MGR_RETIRE_PAGE_INTERVAL  100
+
+static void ras_process_retire_page_dwork(struct work_struct *work)
+{
+       struct amdgpu_ras_mgr *ras_mgr =
+               container_of(work, struct amdgpu_ras_mgr, 
retire_page_dwork.work);
+       struct amdgpu_device *adev = ras_mgr->adev;
+       int ret;
+
+       if (amdgpu_ras_is_rma(adev))
+               return;
+
+       /* If gpu reset is ongoing, delay retiring the bad pages */
+       if (amdgpu_in_reset(adev) || amdgpu_ras_in_recovery(adev)) {
+               schedule_delayed_work(&ras_mgr->retire_page_dwork,
+                       msecs_to_jiffies(RAS_MGR_RETIRE_PAGE_INTERVAL * 3));
+               return;
+       }
+
+       ret = ras_umc_handle_bad_pages(ras_mgr->ras_core, NULL);
+       if (!ret)
+               schedule_delayed_work(&ras_mgr->retire_page_dwork,
+                       msecs_to_jiffies(RAS_MGR_RETIRE_PAGE_INTERVAL));
+}
+
+int amdgpu_ras_process_init(struct amdgpu_device *adev)
+{
+       struct amdgpu_ras_mgr *ras_mgr = amdgpu_ras_mgr_get_context(adev);
+
+       INIT_DELAYED_WORK(&ras_mgr->retire_page_dwork, 
ras_process_retire_page_dwork);
+
+       return 0;
+}
+
+int amdgpu_ras_process_fini(struct amdgpu_device *adev)
+{
+       struct amdgpu_ras_mgr *ras_mgr = amdgpu_ras_mgr_get_context(adev);
+
+       /* Save all cached bad pages to eeprom */
+       flush_delayed_work(&ras_mgr->retire_page_dwork);
+       cancel_delayed_work_sync(&ras_mgr->retire_page_dwork);
+       return 0;
+}
+
+int amdgpu_ras_process_handle_umc_interrupt(struct amdgpu_device *adev, void 
*data)
+{
+       struct amdgpu_ras_mgr *ras_mgr = amdgpu_ras_mgr_get_context(adev);
+
+       if (!ras_mgr->ras_core)
+               return -EINVAL;
+
+       return ras_process_add_interrupt_req(ras_mgr->ras_core, NULL, true);
+}
+
+int amdgpu_ras_process_handle_unexpected_interrupt(struct amdgpu_device *adev, 
void *data)
+{
+       amdgpu_ras_set_fed(adev, true);
+       return amdgpu_ras_mgr_reset_gpu(adev, AMDGPU_RAS_GPU_RESET_MODE1_RESET);
+}
+
+int amdgpu_ras_process_handle_consumption_interrupt(struct amdgpu_device 
*adev, void *data)
+{
+       struct amdgpu_ras_mgr *ras_mgr = amdgpu_ras_mgr_get_context(adev);
+       struct ras_ih_info *ih_info = (struct ras_ih_info *)data;
+       struct ras_event_req req;
+       uint64_t seqno;
+
+       if (!ih_info)
+               return -EINVAL;
+
+       memset(&req, 0, sizeof(req));
+       req.block = ih_info->block;
+       req.data = ih_info->data;
+       req.pasid = ih_info->pasid;
+       req.pasid_fn = ih_info->pasid_fn;
+       req.reset = ih_info->reset;
+
+       seqno = ras_core_get_seqno(ras_mgr->ras_core,
+                               RAS_SEQNO_TYPE_POISON_CONSUMPTION, false);
+
+       /* When the ACA register cannot be read from FW, the poison
+        * consumption seqno in the fifo will not pop up, so it is
+        * necessary to check whether the seqno is the previous seqno.
+        */
+       if (seqno == ras_mgr->last_poison_consumption_seqno) {
+               /* Pop and discard the previous seqno */
+               ras_core_get_seqno(ras_mgr->ras_core,
+                               RAS_SEQNO_TYPE_POISON_CONSUMPTION, true);
+               seqno = ras_core_get_seqno(ras_mgr->ras_core,
+                                       RAS_SEQNO_TYPE_POISON_CONSUMPTION, 
false);
+       }
+       ras_mgr->last_poison_consumption_seqno = seqno;
+       req.seqno = seqno;
+
+       return ras_process_add_interrupt_req(ras_mgr->ras_core, &req, false);
+}
diff --git a/drivers/gpu/drm/amd/ras/ras_mgr/amdgpu_ras_process.h 
b/drivers/gpu/drm/amd/ras/ras_mgr/amdgpu_ras_process.h
new file mode 100644
index 000000000000..b9502bd21beb
--- /dev/null
+++ b/drivers/gpu/drm/amd/ras/ras_mgr/amdgpu_ras_process.h
@@ -0,0 +1,37 @@
+/* SPDX-License-Identifier: MIT */
+/*
+ * Copyright (c) 2025 Advanced Micro Devices, Inc.
+ *
+ * Permission is hereby granted, free of charge, to any person obtaining a copy
+ * of this software and associated documentation files (the "Software"), to 
deal
+ * in the Software without restriction, including without limitation the rights
+ * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+ * copies of the Software, and to permit persons to whom the Software is
+ * furnished to do so, subject to the following conditions:
+ *
+ * The above copyright notice and this permission notice shall be included in
+ * all copies or substantial portions of the Software.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+ * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+ * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL THE
+ * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+ * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING 
FROM,
+ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
+ * THE SOFTWARE.
+ */
+#ifndef __AMDGPU_RAS_PROCESS_H__
+#define __AMDGPU_RAS_PROCESS_H__
+#include "ras_process.h"
+#include "amdgpu_ras_mgr.h"
+
+enum ras_ih_type;
+int amdgpu_ras_process_init(struct amdgpu_device *adev);
+int amdgpu_ras_process_fini(struct amdgpu_device *adev);
+int amdgpu_ras_process_handle_umc_interrupt(struct amdgpu_device *adev,
+               void *data);
+int amdgpu_ras_process_handle_unexpected_interrupt(struct amdgpu_device *adev,
+               void *data);
+int amdgpu_ras_process_handle_consumption_interrupt(struct amdgpu_device *adev,
+               void *data);
+#endif
-- 
2.34.1

Reply via email to