diff options
| author | Andrey Grodzovsky <Andrey.Grodzovsky@amd.com> | 2017-10-19 14:29:46 -0400 |
|---|---|---|
| committer | Alex Deucher <alexander.deucher@amd.com> | 2017-12-04 16:33:08 -0500 |
| commit | d1f6dc1a9a106a73510181cfad9b4a7a0b140990 (patch) | |
| tree | d596b7bcfbac180a1d478dd3bebba45dc56ea404 /drivers | |
| parent | ca797d29cd63e7b71b4eea29aff3b1cefd1ecb59 (diff) | |
drm/amdgpu: Avoid accessing job->entity after the job is scheduled.
Bug: amdgpu_job_free_cb was accessing s_job->s_entity when the allocated
amdgpu_ctx (and the entity inside it) were already deallocated from
amdgpu_cs_parser_fini.
Fix: Save job's priority on it's creation instead of accessing it from
s_entity later on.
Signed-off-by: Andrey Grodzovsky <Andrey.Grodzovsky@amd.com>
Reviewed-by: Andres Rodriguez <andresx7@gmail.com>
Signed-off-by: Alex Deucher <alexander.deucher@amd.com>
Diffstat (limited to 'drivers')
| -rw-r--r-- | drivers/gpu/drm/amd/amdgpu/amdgpu_cs.c | 3 | ||||
| -rw-r--r-- | drivers/gpu/drm/amd/amdgpu/amdgpu_job.c | 5 | ||||
| -rw-r--r-- | drivers/gpu/drm/amd/scheduler/gpu_scheduler.c | 1 | ||||
| -rw-r--r-- | drivers/gpu/drm/amd/scheduler/gpu_scheduler.h | 32 |
4 files changed, 18 insertions, 23 deletions
diff --git a/drivers/gpu/drm/amd/amdgpu/amdgpu_cs.c b/drivers/gpu/drm/amd/amdgpu/amdgpu_cs.c index 57abf7abd7a9..b4b3ecf9f96d 100644 --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_cs.c +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_cs.c | |||
| @@ -1194,8 +1194,7 @@ static int amdgpu_cs_submit(struct amdgpu_cs_parser *p, | |||
| 1194 | job->uf_sequence = seq; | 1194 | job->uf_sequence = seq; |
| 1195 | 1195 | ||
| 1196 | amdgpu_job_free_resources(job); | 1196 | amdgpu_job_free_resources(job); |
| 1197 | amdgpu_ring_priority_get(job->ring, | 1197 | amdgpu_ring_priority_get(job->ring, job->base.s_priority); |
| 1198 | amd_sched_get_job_priority(&job->base)); | ||
| 1199 | 1198 | ||
| 1200 | trace_amdgpu_cs_ioctl(job); | 1199 | trace_amdgpu_cs_ioctl(job); |
| 1201 | amd_sched_entity_push_job(&job->base); | 1200 | amd_sched_entity_push_job(&job->base); |
diff --git a/drivers/gpu/drm/amd/amdgpu/amdgpu_job.c b/drivers/gpu/drm/amd/amdgpu/amdgpu_job.c index 0cfc68db575b..a58e3c5dd84b 100644 --- a/drivers/gpu/drm/amd/amdgpu/amdgpu_job.c +++ b/drivers/gpu/drm/amd/amdgpu/amdgpu_job.c | |||
| @@ -104,7 +104,7 @@ static void amdgpu_job_free_cb(struct amd_sched_job *s_job) | |||
| 104 | { | 104 | { |
| 105 | struct amdgpu_job *job = container_of(s_job, struct amdgpu_job, base); | 105 | struct amdgpu_job *job = container_of(s_job, struct amdgpu_job, base); |
| 106 | 106 | ||
| 107 | amdgpu_ring_priority_put(job->ring, amd_sched_get_job_priority(s_job)); | 107 | amdgpu_ring_priority_put(job->ring, s_job->s_priority); |
| 108 | dma_fence_put(job->fence); | 108 | dma_fence_put(job->fence); |
| 109 | amdgpu_sync_free(&job->sync); | 109 | amdgpu_sync_free(&job->sync); |
| 110 | amdgpu_sync_free(&job->dep_sync); | 110 | amdgpu_sync_free(&job->dep_sync); |
| @@ -141,8 +141,7 @@ int amdgpu_job_submit(struct amdgpu_job *job, struct amdgpu_ring *ring, | |||
| 141 | job->fence_ctx = entity->fence_context; | 141 | job->fence_ctx = entity->fence_context; |
| 142 | *f = dma_fence_get(&job->base.s_fence->finished); | 142 | *f = dma_fence_get(&job->base.s_fence->finished); |
| 143 | amdgpu_job_free_resources(job); | 143 | amdgpu_job_free_resources(job); |
| 144 | amdgpu_ring_priority_get(job->ring, | 144 | amdgpu_ring_priority_get(job->ring, job->base.s_priority); |
| 145 | amd_sched_get_job_priority(&job->base)); | ||
| 146 | amd_sched_entity_push_job(&job->base); | 145 | amd_sched_entity_push_job(&job->base); |
| 147 | 146 | ||
| 148 | return 0; | 147 | return 0; |
diff --git a/drivers/gpu/drm/amd/scheduler/gpu_scheduler.c b/drivers/gpu/drm/amd/scheduler/gpu_scheduler.c index 92ec663fdada..9061c88e3e0c 100644 --- a/drivers/gpu/drm/amd/scheduler/gpu_scheduler.c +++ b/drivers/gpu/drm/amd/scheduler/gpu_scheduler.c | |||
| @@ -529,6 +529,7 @@ int amd_sched_job_init(struct amd_sched_job *job, | |||
| 529 | { | 529 | { |
| 530 | job->sched = sched; | 530 | job->sched = sched; |
| 531 | job->s_entity = entity; | 531 | job->s_entity = entity; |
| 532 | job->s_priority = entity->rq - sched->sched_rq; | ||
| 532 | job->s_fence = amd_sched_fence_create(entity, owner); | 533 | job->s_fence = amd_sched_fence_create(entity, owner); |
| 533 | if (!job->s_fence) | 534 | if (!job->s_fence) |
| 534 | return -ENOMEM; | 535 | return -ENOMEM; |
diff --git a/drivers/gpu/drm/amd/scheduler/gpu_scheduler.h b/drivers/gpu/drm/amd/scheduler/gpu_scheduler.h index 52c8e5447624..3f75b45ed958 100644 --- a/drivers/gpu/drm/amd/scheduler/gpu_scheduler.h +++ b/drivers/gpu/drm/amd/scheduler/gpu_scheduler.h | |||
| @@ -30,6 +30,19 @@ | |||
| 30 | struct amd_gpu_scheduler; | 30 | struct amd_gpu_scheduler; |
| 31 | struct amd_sched_rq; | 31 | struct amd_sched_rq; |
| 32 | 32 | ||
| 33 | enum amd_sched_priority { | ||
| 34 | AMD_SCHED_PRIORITY_MIN, | ||
| 35 | AMD_SCHED_PRIORITY_LOW = AMD_SCHED_PRIORITY_MIN, | ||
| 36 | AMD_SCHED_PRIORITY_NORMAL, | ||
| 37 | AMD_SCHED_PRIORITY_HIGH_SW, | ||
| 38 | AMD_SCHED_PRIORITY_HIGH_HW, | ||
| 39 | AMD_SCHED_PRIORITY_KERNEL, | ||
| 40 | AMD_SCHED_PRIORITY_MAX, | ||
| 41 | AMD_SCHED_PRIORITY_INVALID = -1, | ||
| 42 | AMD_SCHED_PRIORITY_UNSET = -2 | ||
| 43 | }; | ||
| 44 | |||
| 45 | |||
| 33 | /** | 46 | /** |
| 34 | * A scheduler entity is a wrapper around a job queue or a group | 47 | * A scheduler entity is a wrapper around a job queue or a group |
| 35 | * of other entities. Entities take turns emitting jobs from their | 48 | * of other entities. Entities take turns emitting jobs from their |
| @@ -83,6 +96,7 @@ struct amd_sched_job { | |||
| 83 | struct delayed_work work_tdr; | 96 | struct delayed_work work_tdr; |
| 84 | uint64_t id; | 97 | uint64_t id; |
| 85 | atomic_t karma; | 98 | atomic_t karma; |
| 99 | enum amd_sched_priority s_priority; | ||
| 86 | }; | 100 | }; |
| 87 | 101 | ||
| 88 | extern const struct dma_fence_ops amd_sched_fence_ops_scheduled; | 102 | extern const struct dma_fence_ops amd_sched_fence_ops_scheduled; |
| @@ -114,18 +128,6 @@ struct amd_sched_backend_ops { | |||
| 114 | void (*free_job)(struct amd_sched_job *sched_job); | 128 | void (*free_job)(struct amd_sched_job *sched_job); |
| 115 | }; | 129 | }; |
| 116 | 130 | ||
| 117 | enum amd_sched_priority { | ||
| 118 | AMD_SCHED_PRIORITY_MIN, | ||
| 119 | AMD_SCHED_PRIORITY_LOW = AMD_SCHED_PRIORITY_MIN, | ||
| 120 | AMD_SCHED_PRIORITY_NORMAL, | ||
| 121 | AMD_SCHED_PRIORITY_HIGH_SW, | ||
| 122 | AMD_SCHED_PRIORITY_HIGH_HW, | ||
| 123 | AMD_SCHED_PRIORITY_KERNEL, | ||
| 124 | AMD_SCHED_PRIORITY_MAX, | ||
| 125 | AMD_SCHED_PRIORITY_INVALID = -1, | ||
| 126 | AMD_SCHED_PRIORITY_UNSET = -2 | ||
| 127 | }; | ||
| 128 | |||
| 129 | /** | 131 | /** |
| 130 | * One scheduler is implemented for each hardware ring | 132 | * One scheduler is implemented for each hardware ring |
| 131 | */ | 133 | */ |
| @@ -176,10 +178,4 @@ bool amd_sched_dependency_optimized(struct dma_fence* fence, | |||
| 176 | struct amd_sched_entity *entity); | 178 | struct amd_sched_entity *entity); |
| 177 | void amd_sched_job_kickout(struct amd_sched_job *s_job); | 179 | void amd_sched_job_kickout(struct amd_sched_job *s_job); |
| 178 | 180 | ||
| 179 | static inline enum amd_sched_priority | ||
| 180 | amd_sched_get_job_priority(struct amd_sched_job *job) | ||
| 181 | { | ||
| 182 | return (job->s_entity->rq - job->sched->sched_rq); | ||
| 183 | } | ||
| 184 | |||
| 185 | #endif | 181 | #endif |
