drm/nv50/crtc: disable flip overlay around scaling mode changes
Prevents EVO getting all angry at us. Signed-off-by: Ben Skeggs <bskeggs@redhat.com>
This commit is contained in:
@@ -38,6 +38,39 @@
|
||||
#include "nouveau_connector.h"
|
||||
#include "nv50_display.h"
|
||||
|
||||
static int
|
||||
nv50_crtc_wait_complete(struct drm_crtc *crtc)
|
||||
{
|
||||
struct drm_device *dev = crtc->dev;
|
||||
struct drm_nouveau_private *dev_priv = dev->dev_private;
|
||||
struct nouveau_timer_engine *ptimer = &dev_priv->engine.timer;
|
||||
struct nv50_display *disp = nv50_display(dev);
|
||||
struct nouveau_channel *evo = disp->master;
|
||||
u64 start;
|
||||
int ret;
|
||||
|
||||
ret = RING_SPACE(evo, 6);
|
||||
if (ret)
|
||||
return ret;
|
||||
BEGIN_RING(evo, 0, 0x0084, 1);
|
||||
OUT_RING (evo, 0x80000000);
|
||||
BEGIN_RING(evo, 0, 0x0080, 1);
|
||||
OUT_RING (evo, 0);
|
||||
BEGIN_RING(evo, 0, 0x0084, 1);
|
||||
OUT_RING (evo, 0x00000000);
|
||||
|
||||
nv_wo32(disp->ntfy, 0x000, 0x00000000);
|
||||
FIRE_RING (evo);
|
||||
|
||||
start = ptimer->read(dev);
|
||||
do {
|
||||
if (nv_ro32(disp->ntfy, 0x000))
|
||||
return 0;
|
||||
} while (ptimer->read(dev) - start < 2000000000ULL);
|
||||
|
||||
return -EBUSY;
|
||||
}
|
||||
|
||||
static void
|
||||
nv50_crtc_lut_load(struct drm_crtc *crtc)
|
||||
{
|
||||
@@ -184,10 +217,11 @@ nv50_crtc_set_scale(struct nouveau_crtc *nv_crtc, int scaling_mode, bool update)
|
||||
{
|
||||
struct nouveau_connector *nv_connector =
|
||||
nouveau_crtc_connector_get(nv_crtc);
|
||||
struct drm_device *dev = nv_crtc->base.dev;
|
||||
struct drm_crtc *crtc = &nv_crtc->base;
|
||||
struct drm_device *dev = crtc->dev;
|
||||
struct nouveau_channel *evo = nv50_display(dev)->master;
|
||||
struct drm_display_mode *native_mode = NULL;
|
||||
struct drm_display_mode *mode = &nv_crtc->base.mode;
|
||||
struct drm_display_mode *mode = &crtc->mode;
|
||||
uint32_t outX, outY, horiz, vert;
|
||||
int ret;
|
||||
|
||||
@@ -231,7 +265,7 @@ nv50_crtc_set_scale(struct nouveau_crtc *nv_crtc, int scaling_mode, bool update)
|
||||
break;
|
||||
}
|
||||
|
||||
ret = RING_SPACE(evo, update ? 7 : 5);
|
||||
ret = RING_SPACE(evo, 5);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
@@ -251,9 +285,9 @@ nv50_crtc_set_scale(struct nouveau_crtc *nv_crtc, int scaling_mode, bool update)
|
||||
OUT_RING(evo, outY << 16 | outX);
|
||||
|
||||
if (update) {
|
||||
BEGIN_RING(evo, 0, NV50_EVO_UPDATE, 1);
|
||||
OUT_RING(evo, 0);
|
||||
FIRE_RING(evo);
|
||||
nv50_display_flip_stop(crtc);
|
||||
nv50_crtc_wait_complete(crtc);
|
||||
nv50_display_flip_next(crtc, crtc->fb, NULL);
|
||||
}
|
||||
|
||||
return 0;
|
||||
@@ -441,39 +475,6 @@ nv50_crtc_dpms(struct drm_crtc *crtc, int mode)
|
||||
{
|
||||
}
|
||||
|
||||
static int
|
||||
nv50_crtc_wait_complete(struct drm_crtc *crtc)
|
||||
{
|
||||
struct drm_device *dev = crtc->dev;
|
||||
struct drm_nouveau_private *dev_priv = dev->dev_private;
|
||||
struct nouveau_timer_engine *ptimer = &dev_priv->engine.timer;
|
||||
struct nv50_display *disp = nv50_display(dev);
|
||||
struct nouveau_channel *evo = disp->master;
|
||||
u64 start;
|
||||
int ret;
|
||||
|
||||
ret = RING_SPACE(evo, 6);
|
||||
if (ret)
|
||||
return ret;
|
||||
BEGIN_RING(evo, 0, 0x0084, 1);
|
||||
OUT_RING (evo, 0x80000000);
|
||||
BEGIN_RING(evo, 0, 0x0080, 1);
|
||||
OUT_RING (evo, 0);
|
||||
BEGIN_RING(evo, 0, 0x0084, 1);
|
||||
OUT_RING (evo, 0x00000000);
|
||||
|
||||
nv_wo32(disp->ntfy, 0x000, 0x00000000);
|
||||
FIRE_RING (evo);
|
||||
|
||||
start = ptimer->read(dev);
|
||||
do {
|
||||
if (nv_ro32(disp->ntfy, 0x000))
|
||||
return 0;
|
||||
} while (ptimer->read(dev) - start < 2000000000ULL);
|
||||
|
||||
return -EBUSY;
|
||||
}
|
||||
|
||||
static void
|
||||
nv50_crtc_prepare(struct drm_crtc *crtc)
|
||||
{
|
||||
|
Reference in New Issue
Block a user