mirror of
https://github.com/torvalds/linux.git
synced 2025-12-07 20:06:24 +00:00
drm/amd/ras: Amdgpu preprocesses ras interrupts
Amdgpu preprocesses ras interrupts. Signed-off-by: YiPeng Chai <YiPeng.Chai@amd.com> Reviewed-by: Tao Zhou <tao.zhou1@amd.com> Reviewed-by: Hawking Zhang <Hawking.Zhang@amd.com> Signed-off-by: Alex Deucher <alexander.deucher@amd.com>
This commit is contained in:
committed by
Alex Deucher
parent
ffdab7f4e5
commit
e221ac6f42
126
drivers/gpu/drm/amd/ras/ras_mgr/amdgpu_ras_process.c
Normal file
126
drivers/gpu/drm/amd/ras/ras_mgr/amdgpu_ras_process.c
Normal file
@@ -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);
|
||||
}
|
||||
37
drivers/gpu/drm/amd/ras/ras_mgr/amdgpu_ras_process.h
Normal file
37
drivers/gpu/drm/amd/ras/ras_mgr/amdgpu_ras_process.h
Normal file
@@ -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
|
||||
Reference in New Issue
Block a user