Commit 2905db27 authored by Boris Brezillon's avatar Boris Brezillon
Browse files

drm/panfrost: Don't reset the GPU on job faults unless we really have to



If we can recover from a fault without a reset there's no reason to
issue one.

v3:
* Drop the mention of Valhall requiring a reset on JOB_BUS_FAULT
* Set the fence error to -EINVAL instead of having per-exception
  error codes

Signed-off-by: default avatarBoris Brezillon <boris.brezillon@collabora.com>
Reviewed-by: default avatarSteven Price <steven.price@arm.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20210630062751.2832545-14-boris.brezillon@collabora.com
parent f9ab9c66
Loading
Loading
Loading
Loading
+9 −0
Original line number Diff line number Diff line
@@ -379,6 +379,15 @@ const char *panfrost_exception_name(u32 exception_code)
	return panfrost_exception_infos[exception_code].name;
}

bool panfrost_exception_needs_reset(const struct panfrost_device *pfdev,
				    u32 exception_code)
{
	/* Right now, none of the GPU we support need a reset, but this
	 * might change.
	 */
	return false;
}

void panfrost_device_reset(struct panfrost_device *pfdev)
{
	panfrost_gpu_soft_reset(pfdev);
+2 −0
Original line number Diff line number Diff line
@@ -245,6 +245,8 @@ enum drm_panfrost_exception_type {
};

const char *panfrost_exception_name(u32 exception_code);
bool panfrost_exception_needs_reset(const struct panfrost_device *pfdev,
				    u32 exception_code);

static inline void
panfrost_device_schedule_reset(struct panfrost_device *pfdev)
+14 −2
Original line number Diff line number Diff line
@@ -482,14 +482,26 @@ static void panfrost_job_handle_irq(struct panfrost_device *pfdev, u32 status)
		job_write(pfdev, JOB_INT_CLEAR, mask);

		if (status & JOB_INT_MASK_ERR(j)) {
			u32 js_status = job_read(pfdev, JS_STATUS(j));

			job_write(pfdev, JS_COMMAND_NEXT(j), JS_COMMAND_NOP);

			dev_err(pfdev->dev, "js fault, js=%d, status=%s, head=0x%x, tail=0x%x",
				j,
				panfrost_exception_name(job_read(pfdev, JS_STATUS(j))),
				panfrost_exception_name(js_status),
				job_read(pfdev, JS_HEAD_LO(j)),
				job_read(pfdev, JS_TAIL_LO(j)));

			/* If we need a reset, signal it to the timeout
			 * handler, otherwise, update the fence error field and
			 * signal the job fence.
			 */
			if (panfrost_exception_needs_reset(pfdev, js_status)) {
				drm_sched_fault(&pfdev->js->queue[j].sched);
			} else {
				dma_fence_set_error(pfdev->jobs[j]->done_fence, -EINVAL);
				status |= JOB_INT_MASK_DONE(j);
			}
		}

		if (status & JOB_INT_MASK_DONE(j)) {