Commit 5740682e authored by Monk Liu's avatar Monk Liu Committed by Alex Deucher

drm/amdgpu:implement new GPU recover(v3)

1,new imple names amdgpu_gpu_recover which gives more hint
on what it does compared with gpu_reset

2,gpu_recover unify bare-metal and SR-IOV, only the asic reset
part is implemented differently

3,gpu_recover will increase hang job karma and mark its entity/context
as guilty if exceeds limit

V2:

4,in scheduler main routine the job from guilty context  will be immedialy
fake signaled after it poped from queue and its fence be set with
"-ECANCELED" error

5,in scheduler recovery routine all jobs from the guilty entity would be
dropped

6,in run_job() routine the real IB submission would be skipped if @skip parameter
equales true or there was VRAM lost occured.

V3:

7,replace deprecated gpu reset, use new gpu recover
Signed-off-by: default avatarMonk Liu <Monk.Liu@amd.com>
Reviewed-by: default avatarChristian König <christian.koenig@amd.com>
Signed-off-by: default avatarAlex Deucher <alexander.deucher@amd.com>
parent 48f05f29
...@@ -178,6 +178,10 @@ extern int amdgpu_cik_support; ...@@ -178,6 +178,10 @@ extern int amdgpu_cik_support;
#define CIK_CURSOR_WIDTH 128 #define CIK_CURSOR_WIDTH 128
#define CIK_CURSOR_HEIGHT 128 #define CIK_CURSOR_HEIGHT 128
/* GPU RESET flags */
#define AMDGPU_RESET_INFO_VRAM_LOST (1 << 0)
#define AMDGPU_RESET_INFO_FULLRESET (1 << 1)
struct amdgpu_device; struct amdgpu_device;
struct amdgpu_ib; struct amdgpu_ib;
struct amdgpu_cs_parser; struct amdgpu_cs_parser;
...@@ -1833,7 +1837,7 @@ amdgpu_get_sdma_instance(struct amdgpu_ring *ring) ...@@ -1833,7 +1837,7 @@ amdgpu_get_sdma_instance(struct amdgpu_ring *ring)
#define amdgpu_psp_check_fw_loading_status(adev, i) (adev)->firmware.funcs->check_fw_loading_status((adev), (i)) #define amdgpu_psp_check_fw_loading_status(adev, i) (adev)->firmware.funcs->check_fw_loading_status((adev), (i))
/* Common functions */ /* Common functions */
int amdgpu_gpu_reset(struct amdgpu_device *adev); int amdgpu_gpu_recover(struct amdgpu_device *adev, struct amdgpu_job* job);
bool amdgpu_need_backup(struct amdgpu_device *adev); bool amdgpu_need_backup(struct amdgpu_device *adev);
void amdgpu_pci_config_reset(struct amdgpu_device *adev); void amdgpu_pci_config_reset(struct amdgpu_device *adev);
bool amdgpu_need_post(struct amdgpu_device *adev); bool amdgpu_need_post(struct amdgpu_device *adev);
......
...@@ -2827,163 +2827,172 @@ static int amdgpu_recover_vram_from_shadow(struct amdgpu_device *adev, ...@@ -2827,163 +2827,172 @@ static int amdgpu_recover_vram_from_shadow(struct amdgpu_device *adev,
return r; return r;
} }
/** /*
* amdgpu_sriov_gpu_reset - reset the asic * amdgpu_reset - reset ASIC/GPU for bare-metal or passthrough
* *
* @adev: amdgpu device pointer * @adev: amdgpu device pointer
* @job: which job trigger hang * @reset_flags: output param tells caller the reset result
* *
* Attempt the reset the GPU if it has hung (all asics). * attempt to do soft-reset or full-reset and reinitialize Asic
* for SRIOV case. * return 0 means successed otherwise failed
* Returns 0 for success or an error on failure. */
*/ static int amdgpu_reset(struct amdgpu_device *adev, uint64_t* reset_flags)
int amdgpu_sriov_gpu_reset(struct amdgpu_device *adev, struct amdgpu_job *job)
{ {
int i, j, r = 0; bool need_full_reset, vram_lost = 0;
int resched; int r;
struct amdgpu_bo *bo, *tmp;
struct amdgpu_ring *ring;
struct dma_fence *fence = NULL, *next = NULL;
mutex_lock(&adev->virt.lock_reset); need_full_reset = amdgpu_need_full_reset(adev);
atomic_inc(&adev->gpu_reset_counter);
adev->in_sriov_reset = true;
/* block TTM */ if (!need_full_reset) {
resched = ttm_bo_lock_delayed_workqueue(&adev->mman.bdev); amdgpu_pre_soft_reset(adev);
r = amdgpu_soft_reset(adev);
amdgpu_post_soft_reset(adev);
if (r || amdgpu_check_soft_reset(adev)) {
DRM_INFO("soft reset failed, will fallback to full reset!\n");
need_full_reset = true;
}
/* we start from the ring trigger GPU hang */ }
j = job ? job->ring->idx : 0;
/* block scheduler */ if (need_full_reset) {
for (i = j; i < j + AMDGPU_MAX_RINGS; ++i) { r = amdgpu_suspend(adev);
ring = adev->rings[i % AMDGPU_MAX_RINGS];
if (!ring || !ring->sched.thread)
continue;
kthread_park(ring->sched.thread); retry:
amdgpu_atombios_scratch_regs_save(adev);
r = amdgpu_asic_reset(adev);
amdgpu_atombios_scratch_regs_restore(adev);
/* post card */
amdgpu_atom_asic_init(adev->mode_info.atom_context);
if (job && j != i) if (!r) {
continue; dev_info(adev->dev, "GPU reset succeeded, trying to resume\n");
r = amdgpu_resume_phase1(adev);
if (r)
goto out;
/* here give the last chance to check if job removed from mirror-list vram_lost = amdgpu_check_vram_lost(adev);
* since we already pay some time on kthread_park */ if (vram_lost) {
if (job && list_empty(&job->base.node)) { DRM_ERROR("VRAM is lost!\n");
kthread_unpark(ring->sched.thread); atomic_inc(&adev->vram_lost_counter);
goto give_up_reset;
} }
if (amd_sched_invalidate_job(&job->base, amdgpu_job_hang_limit)) r = amdgpu_ttm_recover_gart(adev);
amd_sched_job_kickout(&job->base); if (r)
goto out;
/* only do job_reset on the hang ring if @job not NULL */ r = amdgpu_resume_phase2(adev);
amd_sched_hw_job_reset(&ring->sched, NULL); if (r)
goto out;
/* after all hw jobs are reset, hw fence is meaningless, so force_completion */ if (vram_lost)
amdgpu_fence_driver_force_completion(ring); amdgpu_fill_reset_magic(adev);
}
} }
/* request to take full control of GPU before re-initialization */ out:
if (job) if (!r) {
amdgpu_virt_reset_gpu(adev); amdgpu_irq_gpu_reset_resume_helper(adev);
else r = amdgpu_ib_ring_tests(adev);
amdgpu_virt_request_full_gpu(adev, true); if (r) {
dev_err(adev->dev, "ib ring test failed (%d).\n", r);
r = amdgpu_suspend(adev);
need_full_reset = true;
goto retry;
}
}
if (reset_flags) {
if (vram_lost)
(*reset_flags) |= AMDGPU_RESET_INFO_VRAM_LOST;
if (need_full_reset)
(*reset_flags) |= AMDGPU_RESET_INFO_FULLRESET;
}
return r;
}
/*
* amdgpu_reset_sriov - reset ASIC for SR-IOV vf
*
* @adev: amdgpu device pointer
* @reset_flags: output param tells caller the reset result
*
* do VF FLR and reinitialize Asic
* return 0 means successed otherwise failed
*/
static int amdgpu_reset_sriov(struct amdgpu_device *adev, uint64_t *reset_flags, bool from_hypervisor)
{
int r;
if (from_hypervisor)
r = amdgpu_virt_request_full_gpu(adev, true);
else
r = amdgpu_virt_reset_gpu(adev);
if (r)
return r;
/* Resume IP prior to SMC */ /* Resume IP prior to SMC */
amdgpu_sriov_reinit_early(adev); r = amdgpu_sriov_reinit_early(adev);
if (r)
goto error;
/* we need recover gart prior to run SMC/CP/SDMA resume */ /* we need recover gart prior to run SMC/CP/SDMA resume */
amdgpu_ttm_recover_gart(adev); amdgpu_ttm_recover_gart(adev);
/* now we are okay to resume SMC/CP/SDMA */ /* now we are okay to resume SMC/CP/SDMA */
amdgpu_sriov_reinit_late(adev); r = amdgpu_sriov_reinit_late(adev);
if (r)
goto error;
amdgpu_irq_gpu_reset_resume_helper(adev); amdgpu_irq_gpu_reset_resume_helper(adev);
r = amdgpu_ib_ring_tests(adev);
if (amdgpu_ib_ring_tests(adev)) if (r)
dev_err(adev->dev, "[GPU_RESET] ib ring test failed (%d).\n", r); dev_err(adev->dev, "[GPU_RESET] ib ring test failed (%d).\n", r);
error:
/* release full control of GPU after ib test */ /* release full control of GPU after ib test */
amdgpu_virt_release_full_gpu(adev, true); amdgpu_virt_release_full_gpu(adev, true);
DRM_INFO("recover vram bo from shadow\n"); if (reset_flags) {
/* will get vram_lost from GIM in future, now all
ring = adev->mman.buffer_funcs_ring; * reset request considered VRAM LOST
mutex_lock(&adev->shadow_list_lock); */
list_for_each_entry_safe(bo, tmp, &adev->shadow_list, shadow_list) { (*reset_flags) |= ~AMDGPU_RESET_INFO_VRAM_LOST;
next = NULL; atomic_inc(&adev->vram_lost_counter);
amdgpu_recover_vram_from_shadow(adev, ring, bo, &next);
if (fence) {
r = dma_fence_wait(fence, false);
if (r) {
WARN(r, "recovery from shadow isn't completed\n");
break;
}
}
dma_fence_put(fence);
fence = next;
}
mutex_unlock(&adev->shadow_list_lock);
if (fence) {
r = dma_fence_wait(fence, false);
if (r)
WARN(r, "recovery from shadow isn't completed\n");
}
dma_fence_put(fence);
for (i = j; i < j + AMDGPU_MAX_RINGS; ++i) {
ring = adev->rings[i % AMDGPU_MAX_RINGS];
if (!ring || !ring->sched.thread)
continue;
if (job && j != i) {
kthread_unpark(ring->sched.thread);
continue;
}
amd_sched_job_recovery(&ring->sched);
kthread_unpark(ring->sched.thread);
}
drm_helper_resume_force_mode(adev->ddev); /* VF FLR or hotlink reset is always full-reset */
give_up_reset: (*reset_flags) |= AMDGPU_RESET_INFO_FULLRESET;
ttm_bo_unlock_delayed_workqueue(&adev->mman.bdev, resched);
if (r) {
/* bad news, how to tell it to userspace ? */
dev_info(adev->dev, "GPU reset failed\n");
} else {
dev_info(adev->dev, "GPU reset successed!\n");
} }
adev->in_sriov_reset = false;
mutex_unlock(&adev->virt.lock_reset);
return r; return r;
} }
/** /**
* amdgpu_gpu_reset - reset the asic * amdgpu_gpu_recover - reset the asic and recover scheduler
* *
* @adev: amdgpu device pointer * @adev: amdgpu device pointer
* @job: which job trigger hang
* *
* Attempt the reset the GPU if it has hung (all asics). * Attempt to reset the GPU if it has hung (all asics).
* Returns 0 for success or an error on failure. * Returns 0 for success or an error on failure.
*/ */
int amdgpu_gpu_reset(struct amdgpu_device *adev) int amdgpu_gpu_recover(struct amdgpu_device *adev, struct amdgpu_job *job)
{ {
struct drm_atomic_state *state = NULL; struct drm_atomic_state *state = NULL;
int i, r; uint64_t reset_flags = 0;
int resched; int i, r, resched;
bool need_full_reset, vram_lost = false;
if (!amdgpu_check_soft_reset(adev)) { if (!amdgpu_check_soft_reset(adev)) {
DRM_INFO("No hardware hang detected. Did some blocks stall?\n"); DRM_INFO("No hardware hang detected. Did some blocks stall?\n");
return 0; return 0;
} }
dev_info(adev->dev, "GPU reset begin!\n");
mutex_lock(&adev->virt.lock_reset);
atomic_inc(&adev->gpu_reset_counter); atomic_inc(&adev->gpu_reset_counter);
adev->in_sriov_reset = 1;
/* block TTM */ /* block TTM */
resched = ttm_bo_lock_delayed_workqueue(&adev->mman.bdev); resched = ttm_bo_lock_delayed_workqueue(&adev->mman.bdev);
...@@ -2997,69 +3006,26 @@ int amdgpu_gpu_reset(struct amdgpu_device *adev) ...@@ -2997,69 +3006,26 @@ int amdgpu_gpu_reset(struct amdgpu_device *adev)
if (!ring || !ring->sched.thread) if (!ring || !ring->sched.thread)
continue; continue;
/* only focus on the ring hit timeout if &job not NULL */
if (job && job->ring->idx != i)
continue;
kthread_park(ring->sched.thread); kthread_park(ring->sched.thread);
amd_sched_hw_job_reset(&ring->sched, NULL); amd_sched_hw_job_reset(&ring->sched, &job->base);
/* after all hw jobs are reset, hw fence is meaningless, so force_completion */ /* after all hw jobs are reset, hw fence is meaningless, so force_completion */
amdgpu_fence_driver_force_completion(ring); amdgpu_fence_driver_force_completion(ring);
} }
need_full_reset = amdgpu_need_full_reset(adev); if (amdgpu_sriov_vf(adev))
r = amdgpu_reset_sriov(adev, &reset_flags, job ? false : true);
if (!need_full_reset) { else
amdgpu_pre_soft_reset(adev); r = amdgpu_reset(adev, &reset_flags);
r = amdgpu_soft_reset(adev);
amdgpu_post_soft_reset(adev);
if (r || amdgpu_check_soft_reset(adev)) {
DRM_INFO("soft reset failed, will fallback to full reset!\n");
need_full_reset = true;
}
}
if (need_full_reset) {
r = amdgpu_suspend(adev);
retry:
amdgpu_atombios_scratch_regs_save(adev);
r = amdgpu_asic_reset(adev);
amdgpu_atombios_scratch_regs_restore(adev);
/* post card */
amdgpu_atom_asic_init(adev->mode_info.atom_context);
if (!r) { if (!r) {
dev_info(adev->dev, "GPU reset succeeded, trying to resume\n"); if (((reset_flags & AMDGPU_RESET_INFO_FULLRESET) && !(adev->flags & AMD_IS_APU)) ||
r = amdgpu_resume_phase1(adev); (reset_flags & AMDGPU_RESET_INFO_VRAM_LOST)) {
if (r)
goto out;
vram_lost = amdgpu_check_vram_lost(adev);
if (vram_lost) {
DRM_ERROR("VRAM is lost!\n");
atomic_inc(&adev->vram_lost_counter);
}
r = amdgpu_ttm_recover_gart(adev);
if (r)
goto out;
r = amdgpu_resume_phase2(adev);
if (r)
goto out;
if (vram_lost)
amdgpu_fill_reset_magic(adev);
}
}
out:
if (!r) {
amdgpu_irq_gpu_reset_resume_helper(adev);
r = amdgpu_ib_ring_tests(adev);
if (r) {
dev_err(adev->dev, "ib ring test failed (%d).\n", r);
r = amdgpu_suspend(adev);
need_full_reset = true;
goto retry;
}
/**
* recovery vm page tables, since we cannot depend on VRAM is
* consistent after gpu full reset.
*/
if (need_full_reset && amdgpu_need_backup(adev)) {
struct amdgpu_ring *ring = adev->mman.buffer_funcs_ring; struct amdgpu_ring *ring = adev->mman.buffer_funcs_ring;
struct amdgpu_bo *bo, *tmp; struct amdgpu_bo *bo, *tmp;
struct dma_fence *fence = NULL, *next = NULL; struct dma_fence *fence = NULL, *next = NULL;
...@@ -3088,40 +3054,56 @@ int amdgpu_gpu_reset(struct amdgpu_device *adev) ...@@ -3088,40 +3054,56 @@ int amdgpu_gpu_reset(struct amdgpu_device *adev)
} }
dma_fence_put(fence); dma_fence_put(fence);
} }
for (i = 0; i < AMDGPU_MAX_RINGS; ++i) { for (i = 0; i < AMDGPU_MAX_RINGS; ++i) {
struct amdgpu_ring *ring = adev->rings[i]; struct amdgpu_ring *ring = adev->rings[i];
if (!ring || !ring->sched.thread) if (!ring || !ring->sched.thread)
continue; continue;
/* only focus on the ring hit timeout if &job not NULL */
if (job && job->ring->idx != i)
continue;
amd_sched_job_recovery(&ring->sched); amd_sched_job_recovery(&ring->sched);
kthread_unpark(ring->sched.thread); kthread_unpark(ring->sched.thread);
} }
} else { } else {
dev_err(adev->dev, "asic resume failed (%d).\n", r);
for (i = 0; i < AMDGPU_MAX_RINGS; ++i) { for (i = 0; i < AMDGPU_MAX_RINGS; ++i) {
if (adev->rings[i] && adev->rings[i]->sched.thread) { struct amdgpu_ring *ring = adev->rings[i];
if (!ring || !ring->sched.thread)
continue;
/* only focus on the ring hit timeout if &job not NULL */
if (job && job->ring->idx != i)
continue;
kthread_unpark(adev->rings[i]->sched.thread); kthread_unpark(adev->rings[i]->sched.thread);
} }
} }
}
if (amdgpu_device_has_dc_support(adev)) { if (amdgpu_device_has_dc_support(adev)) {
r = drm_atomic_helper_resume(adev->ddev, state); if (drm_atomic_helper_resume(adev->ddev, state))
dev_info(adev->dev, "drm resume failed:%d\n", r);
amdgpu_dm_display_resume(adev); amdgpu_dm_display_resume(adev);
} else } else {
drm_helper_resume_force_mode(adev->ddev); drm_helper_resume_force_mode(adev->ddev);
}
ttm_bo_unlock_delayed_workqueue(&adev->mman.bdev, resched); ttm_bo_unlock_delayed_workqueue(&adev->mman.bdev, resched);
if (r) { if (r) {
/* bad news, how to tell it to userspace ? */ /* bad news, how to tell it to userspace ? */
dev_info(adev->dev, "GPU reset failed\n"); dev_info(adev->dev, "GPU reset(%d) failed\n", atomic_read(&adev->gpu_reset_counter));
} amdgpu_vf_error_put(adev, AMDGIM_ERROR_VF_GPU_RESET_FAIL, 0, r);
else { } else {
dev_info(adev->dev, "GPU reset successed!\n"); dev_info(adev->dev, "GPU reset(%d) successed!\n",atomic_read(&adev->gpu_reset_counter));
} }
amdgpu_vf_error_trans_all(adev); amdgpu_vf_error_trans_all(adev);
adev->in_sriov_reset = 0;
mutex_unlock(&adev->virt.lock_reset);
return r; return r;
} }
......
...@@ -694,25 +694,25 @@ static int amdgpu_debugfs_fence_info(struct seq_file *m, void *data) ...@@ -694,25 +694,25 @@ static int amdgpu_debugfs_fence_info(struct seq_file *m, void *data)
} }
/** /**
* amdgpu_debugfs_gpu_reset - manually trigger a gpu reset * amdgpu_debugfs_gpu_recover - manually trigger a gpu reset & recover
* *
* Manually trigger a gpu reset at the next fence wait. * Manually trigger a gpu reset at the next fence wait.
*/ */
static int amdgpu_debugfs_gpu_reset(struct seq_file *m, void *data) static int amdgpu_debugfs_gpu_recover(struct seq_file *m, void *data)
{ {
struct drm_info_node *node = (struct drm_info_node *) m->private; struct drm_info_node *node = (struct drm_info_node *) m->private;
struct drm_device *dev = node->minor->dev; struct drm_device *dev = node->minor->dev;
struct amdgpu_device *adev = dev->dev_private; struct amdgpu_device *adev = dev->dev_private;
seq_printf(m, "gpu reset\n"); seq_printf(m, "gpu recover\n");
amdgpu_gpu_reset(adev); amdgpu_gpu_recover(adev, NULL);
return 0; return 0;
} }
static const struct drm_info_list amdgpu_debugfs_fence_list[] = { static const struct drm_info_list amdgpu_debugfs_fence_list[] = {
{"amdgpu_fence_info", &amdgpu_debugfs_fence_info, 0, NULL}, {"amdgpu_fence_info", &amdgpu_debugfs_fence_info, 0, NULL},
{"amdgpu_gpu_reset", &amdgpu_debugfs_gpu_reset, 0, NULL} {"amdgpu_gpu_recover", &amdgpu_debugfs_gpu_recover, 0, NULL}
}; };
static const struct drm_info_list amdgpu_debugfs_fence_list_sriov[] = { static const struct drm_info_list amdgpu_debugfs_fence_list_sriov[] = {
......
...@@ -88,7 +88,7 @@ static void amdgpu_irq_reset_work_func(struct work_struct *work) ...@@ -88,7 +88,7 @@ static void amdgpu_irq_reset_work_func(struct work_struct *work)
reset_work); reset_work);
if (!amdgpu_sriov_vf(adev)) if (!amdgpu_sriov_vf(adev))
amdgpu_gpu_reset(adev); amdgpu_gpu_recover(adev, NULL);
} }
/* Disable *all* interrupts */ /* Disable *all* interrupts */
......
...@@ -37,10 +37,7 @@ static void amdgpu_job_timedout(struct amd_sched_job *s_job) ...@@ -37,10 +37,7 @@ static void amdgpu_job_timedout(struct amd_sched_job *s_job)
atomic_read(&job->ring->fence_drv.last_seq), atomic_read(&job->ring->fence_drv.last_seq),
job->ring->fence_drv.sync_seq); job->ring->fence_drv.sync_seq);
if (amdgpu_sriov_vf(job->adev)) amdgpu_gpu_recover(job->adev, job);
amdgpu_sriov_gpu_reset(job->adev, job);
else
amdgpu_gpu_reset(job->adev);
} }
int amdgpu_job_alloc(struct amdgpu_device *adev, unsigned num_ibs, int amdgpu_job_alloc(struct amdgpu_device *adev, unsigned num_ibs,
......
...@@ -288,7 +288,6 @@ int amdgpu_virt_request_full_gpu(struct amdgpu_device *adev, bool init); ...@@ -288,7 +288,6 @@ int amdgpu_virt_request_full_gpu(struct amdgpu_device *adev, bool init);
int amdgpu_virt_release_full_gpu(struct amdgpu_device *adev, bool init); int amdgpu_virt_release_full_gpu(struct amdgpu_device *adev, bool init);
int amdgpu_virt_reset_gpu(struct amdgpu_device *adev); int amdgpu_virt_reset_gpu(struct amdgpu_device *adev);
int amdgpu_virt_wait_reset(struct amdgpu_device *adev); int amdgpu_virt_wait_reset(struct amdgpu_device *adev);
int amdgpu_sriov_gpu_reset(struct amdgpu_device *adev, struct amdgpu_job *job);
int amdgpu_virt_alloc_mm_table(struct amdgpu_device *adev); int amdgpu_virt_alloc_mm_table(struct amdgpu_device *adev);
void amdgpu_virt_free_mm_table(struct amdgpu_device *adev); void amdgpu_virt_free_mm_table(struct amdgpu_device *adev);
int amdgpu_virt_fw_reserve_get_checksum(void *obj, unsigned long obj_size, int amdgpu_virt_fw_reserve_get_checksum(void *obj, unsigned long obj_size,
......
...@@ -254,7 +254,7 @@ static void xgpu_ai_mailbox_flr_work(struct work_struct *work) ...@@ -254,7 +254,7 @@ static void xgpu_ai_mailbox_flr_work(struct work_struct *work)
} }
/* Trigger recovery due to world switch failure */ /* Trigger recovery due to world switch failure */
amdgpu_sriov_gpu_reset(adev, NULL); amdgpu_gpu_recover(adev, NULL);
} }
static int xgpu_ai_set_mailbox_rcv_irq(struct amdgpu_device *adev, static int xgpu_ai_set_mailbox_rcv_irq(struct amdgpu_device *adev,
......
...@@ -519,7 +519,7 @@ static void xgpu_vi_mailbox_flr_work(struct work_struct *work) ...@@ -519,7 +519,7 @@ static void xgpu_vi_mailbox_flr_work(struct work_struct *work)
} }
/* Trigger recovery due to world switch failure */ /* Trigger recovery due to world switch failure */
amdgpu_sriov_gpu_reset(adev, NULL); amdgpu_gpu_recover(adev, NULL);
} }
static int xgpu_vi_set_mailbox_rcv_irq(struct amdgpu_device *adev, static int xgpu_vi_set_mailbox_rcv_irq(struct amdgpu_device *adev,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment