Commit c4ce60dc authored by Eric Anholt's avatar Eric Anholt

drm/vc4: Fix spurious GPU resets due to BO reuse.

We were tracking the "where are the head pointers pointing" globally,
so if another job reused the same BOs and execution was at the same
point as last time we checked, we'd stop and trigger a reset even
though the GPU had made progress.
Signed-off-by: default avatarEric Anholt <eric@anholt.net>
parent 2ee94657
...@@ -92,7 +92,6 @@ struct vc4_dev { ...@@ -92,7 +92,6 @@ struct vc4_dev {
struct work_struct overflow_mem_work; struct work_struct overflow_mem_work;
struct { struct {
uint32_t last_ct0ca, last_ct1ca;
struct timer_list timer; struct timer_list timer;
struct work_struct reset_work; struct work_struct reset_work;
} hangcheck; } hangcheck;
...@@ -192,6 +191,11 @@ struct vc4_exec_info { ...@@ -192,6 +191,11 @@ struct vc4_exec_info {
/* Sequence number for this bin/render job. */ /* Sequence number for this bin/render job. */
uint64_t seqno; uint64_t seqno;
/* Last current addresses the hardware was processing when the
* hangcheck timer checked on us.
*/
uint32_t last_ct0ca, last_ct1ca;
/* Kernel-space copy of the ioctl arguments */ /* Kernel-space copy of the ioctl arguments */
struct drm_vc4_submit_cl *args; struct drm_vc4_submit_cl *args;
......
...@@ -257,10 +257,17 @@ vc4_hangcheck_elapsed(unsigned long data) ...@@ -257,10 +257,17 @@ vc4_hangcheck_elapsed(unsigned long data)
struct drm_device *dev = (struct drm_device *)data; struct drm_device *dev = (struct drm_device *)data;
struct vc4_dev *vc4 = to_vc4_dev(dev); struct vc4_dev *vc4 = to_vc4_dev(dev);
uint32_t ct0ca, ct1ca; uint32_t ct0ca, ct1ca;
unsigned long irqflags;
struct vc4_exec_info *exec;
spin_lock_irqsave(&vc4->job_lock, irqflags);
exec = vc4_first_job(vc4);
/* If idle, we can stop watching for hangs. */ /* If idle, we can stop watching for hangs. */
if (list_empty(&vc4->job_list)) if (!exec) {
spin_unlock_irqrestore(&vc4->job_lock, irqflags);
return; return;
}
ct0ca = V3D_READ(V3D_CTNCA(0)); ct0ca = V3D_READ(V3D_CTNCA(0));
ct1ca = V3D_READ(V3D_CTNCA(1)); ct1ca = V3D_READ(V3D_CTNCA(1));
...@@ -268,14 +275,16 @@ vc4_hangcheck_elapsed(unsigned long data) ...@@ -268,14 +275,16 @@ vc4_hangcheck_elapsed(unsigned long data)
/* If we've made any progress in execution, rearm the timer /* If we've made any progress in execution, rearm the timer
* and wait. * and wait.
*/ */
if (ct0ca != vc4->hangcheck.last_ct0ca || if (ct0ca != exec->last_ct0ca || ct1ca != exec->last_ct1ca) {
ct1ca != vc4->hangcheck.last_ct1ca) { exec->last_ct0ca = ct0ca;
vc4->hangcheck.last_ct0ca = ct0ca; exec->last_ct1ca = ct1ca;
vc4->hangcheck.last_ct1ca = ct1ca; spin_unlock_irqrestore(&vc4->job_lock, irqflags);
vc4_queue_hangcheck(dev); vc4_queue_hangcheck(dev);
return; return;
} }
spin_unlock_irqrestore(&vc4->job_lock, irqflags);
/* We've gone too long with no progress, reset. This has to /* We've gone too long with no progress, reset. This has to
* be done from a work struct, since resetting can sleep and * be done from a work struct, since resetting can sleep and
* this timer hook isn't allowed to. * this timer hook isn't allowed to.
......
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