drm/nouveau/kms/gv100-: fix spurious window immediate interlocks

Cursor position updates were accidentally causing us to attempt to interlock
window with window immediate, and without a matching window immediate update,
NVDisplay could hang forever in some circumstances.

Fixes suspend/resume on (at least) Quadro RTX4000 (TU104).

Reported-by: Lyude Paul <lyude@redhat.com>
Signed-off-by: Ben Skeggs <bskeggs@redhat.com>
This commit is contained in:
Ben Skeggs 2019-05-03 12:23:55 +10:00
parent 30df16b93b
commit d2434e4d94
3 changed files with 3 additions and 1 deletions

View File

@ -41,6 +41,7 @@ struct nv50_disp_interlock {
NV50_DISP_INTERLOCK__SIZE NV50_DISP_INTERLOCK__SIZE
} type; } type;
u32 data; u32 data;
u32 wimm;
}; };
void corec37d_ntfy_init(struct nouveau_bo *, u32); void corec37d_ntfy_init(struct nouveau_bo *, u32);

View File

@ -75,6 +75,7 @@ wimmc37b_init_(const struct nv50_wimm_func *func, struct nouveau_drm *drm,
return ret; return ret;
} }
wndw->interlock.wimm = wndw->interlock.data;
wndw->immd = func; wndw->immd = func;
return 0; return 0;
} }

View File

@ -149,7 +149,7 @@ nv50_wndw_flush_set(struct nv50_wndw *wndw, u32 *interlock,
if (asyw->set.point) { if (asyw->set.point) {
if (asyw->set.point = false, asyw->set.mask) if (asyw->set.point = false, asyw->set.mask)
interlock[wndw->interlock.type] |= wndw->interlock.data; interlock[wndw->interlock.type] |= wndw->interlock.data;
interlock[NV50_DISP_INTERLOCK_WIMM] |= wndw->interlock.data; interlock[NV50_DISP_INTERLOCK_WIMM] |= wndw->interlock.wimm;
wndw->immd->point(wndw, asyw); wndw->immd->point(wndw, asyw);
wndw->immd->update(wndw, interlock); wndw->immd->update(wndw, interlock);