Scaling code didn't took into account that doublescan modes
actually are physically 2x verical resolution, thus scaler needs
to scale logical resolution 2x
Also remove stray OUT_RING that just by a chance didn't cause problems

Signed-off-by: Maxim Levitsky <[email protected]>
---
 drivers/gpu/drm/nouveau/nv50_crtc.c |   21 ++++++++++++++-------
 1 files changed, 14 insertions(+), 7 deletions(-)

diff --git a/drivers/gpu/drm/nouveau/nv50_crtc.c 
b/drivers/gpu/drm/nouveau/nv50_crtc.c
index 882080e..d1315ce 100644
--- a/drivers/gpu/drm/nouveau/nv50_crtc.c
+++ b/drivers/gpu/drm/nouveau/nv50_crtc.c
@@ -235,12 +235,14 @@ nv50_crtc_set_scale(struct nouveau_crtc *nv_crtc, int 
scaling_mode, bool update)
        if (ret)
                return ret;
 
+       if (mode->flags & DRM_MODE_FLAG_DBLSCAN)
+               outY *= 2;
+
        /* Got a better name for SCALER_ACTIVE? */
        /* One day i've got to really figure out why this is needed. */
        BEGIN_RING(evo, 0, NV50_EVO_CRTC(nv_crtc->index, SCALE_CTRL), 1);
-       if ((mode->flags & DRM_MODE_FLAG_DBLSCAN) ||
-           (mode->flags & DRM_MODE_FLAG_INTERLACE) ||
-           mode->hdisplay != outX || mode->vdisplay != outY) {
+       if (mode->flags & DRM_MODE_FLAG_INTERLACE ||
+               mode->hdisplay != outX || mode->vdisplay != outY) {
                OUT_RING(evo, NV50_EVO_CRTC_SCALE_CTRL_ACTIVE);
        } else {
                OUT_RING(evo, NV50_EVO_CRTC_SCALE_CTRL_INACTIVE);
@@ -603,6 +605,7 @@ nv50_crtc_mode_set(struct drm_crtc *crtc, struct 
drm_display_mode *mode,
        struct nouveau_connector *nv_connector = NULL;
        uint32_t hsync_dur,  vsync_dur, hsync_start_to_end, vsync_start_to_end;
        uint32_t hunk1, vunk1, vunk2a, vunk2b;
+       uint32_t vtotal, htotal;
        int ret;
 
        /* Find the connector attached to this CRTC */
@@ -626,6 +629,8 @@ nv50_crtc_mode_set(struct drm_crtc *crtc, struct 
drm_display_mode *mode,
                 adjusted_mode->vsync_start + adjusted_mode->vdisplay;
        vunk2b = adjusted_mode->vtotal -
                 adjusted_mode->vsync_start + adjusted_mode->vtotal;
+       vtotal = adjusted_mode->vtotal;
+       htotal = adjusted_mode->htotal;
 
        if (adjusted_mode->flags & DRM_MODE_FLAG_INTERLACE) {
                vsync_dur /= 2;
@@ -640,6 +645,11 @@ nv50_crtc_mode_set(struct drm_crtc *crtc, struct 
drm_display_mode *mode,
                        vunk2a -= 1;
                        vunk2b -= 1;
                }
+       } else if (adjusted_mode->flags & DRM_MODE_FLAG_DBLSCAN) {
+               vtotal *= 2;
+               vsync_dur *= 2;
+               vsync_start_to_end  *= 2;
+               vunk1 *= 2;
        }
 
        ret = RING_SPACE(evo, 17);
@@ -652,7 +662,7 @@ nv50_crtc_mode_set(struct drm_crtc *crtc, struct 
drm_display_mode *mode,
 
        BEGIN_RING(evo, 0, NV50_EVO_CRTC(nv_crtc->index, DISPLAY_START), 5);
        OUT_RING(evo, 0);
-       OUT_RING(evo, (adjusted_mode->vtotal << 16) | adjusted_mode->htotal);
+       OUT_RING(evo, (vtotal << 16) | htotal);
        OUT_RING(evo, (vsync_dur - 1) << 16 | (hsync_dur - 1));
        OUT_RING(evo, (vsync_start_to_end - 1) << 16 |
                        (hsync_start_to_end - 1));
@@ -661,9 +671,6 @@ nv50_crtc_mode_set(struct drm_crtc *crtc, struct 
drm_display_mode *mode,
        if (adjusted_mode->flags & DRM_MODE_FLAG_INTERLACE) {
                BEGIN_RING(evo, 0, NV50_EVO_CRTC(nv_crtc->index, UNK0824), 1);
                OUT_RING(evo, (vunk2b - 1) << 16 | (vunk2a - 1));
-       } else {
-               OUT_RING(evo, 0);
-               OUT_RING(evo, 0);
        }
 
        BEGIN_RING(evo, 0, NV50_EVO_CRTC(nv_crtc->index, UNK082C), 1);
-- 
1.7.4.1

_______________________________________________
Nouveau mailing list
[email protected]
http://lists.freedesktop.org/mailman/listinfo/nouveau

Reply via email to