diff options
Diffstat (limited to 'drivers/gpu/drm/amd/amdgpu/amdgpu_job.c')
| -rw-r--r-- | drivers/gpu/drm/amd/amdgpu/amdgpu_job.c | 38 | 
1 files changed, 38 insertions, 0 deletions
diff --git a/drivers/gpu/drm/amd/amdgpu/amdgpu_job.c b/drivers/gpu/drm/amd/amdgpu/amdgpu_job.c index 96b2a31ccfed..4fb20e870e63 100644 --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_job.c +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_job.c @@ -248,6 +248,44 @@ static struct dma_fence *amdgpu_job_run(struct drm_sched_job *sched_job)  	return fence;  } +#define to_drm_sched_job(sched_job)		\ +		container_of((sched_job), struct drm_sched_job, queue_node) + +void amdgpu_job_stop_all_jobs_on_sched(struct drm_gpu_scheduler *sched) +{ +	struct drm_sched_job *s_job; +	struct drm_sched_entity *s_entity = NULL; +	int i; + +	/* Signal all jobs not yet scheduled */ +	for (i = DRM_SCHED_PRIORITY_MAX - 1; i >= DRM_SCHED_PRIORITY_MIN; i--) { +		struct drm_sched_rq *rq = &sched->sched_rq[i]; + +		if (!rq) +			continue; + +		spin_lock(&rq->lock); +		list_for_each_entry(s_entity, &rq->entities, list) { +			while ((s_job = to_drm_sched_job(spsc_queue_pop(&s_entity->job_queue)))) { +				struct drm_sched_fence *s_fence = s_job->s_fence; + +				dma_fence_signal(&s_fence->scheduled); +				dma_fence_set_error(&s_fence->finished, -EHWPOISON); +				dma_fence_signal(&s_fence->finished); +			} +		} +		spin_unlock(&rq->lock); +	} + +	/* Signal all jobs already scheduled to HW */ +	list_for_each_entry(s_job, &sched->ring_mirror_list, node) { +		struct drm_sched_fence *s_fence = s_job->s_fence; + +		dma_fence_set_error(&s_fence->finished, -EHWPOISON); +		dma_fence_signal(&s_fence->finished); +	} +} +  const struct drm_sched_backend_ops amdgpu_sched_ops = {  	.dependency = amdgpu_job_dependency,  	.run_job = amdgpu_job_run,  |