Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/sys/dev/pci Also set the bits for horizontal sync start dela...



details:   https://anonhg.NetBSD.org/src/rev/3c91aabcd9ac
branches:  trunk
changeset: 974651:3c91aabcd9ac
user:      jdc <jdc%NetBSD.org@localhost>
date:      Fri Aug 07 18:26:33 2020 +0000

description:
Also set the bits for horizontal sync start delay.  This makes machfb work
correctly for some modes (e.g. 1152x720).
Modify mach64_get_mode() to get the mode from the chip registers (only used
with debug on).
Whilst here tidy up some other debug output.

diffstat:

 sys/dev/pci/machfb.c |  169 ++++++++++++++++++++++++++++++++++++++------------
 1 files changed, 126 insertions(+), 43 deletions(-)

diffs (263 lines):

diff -r 7f23f181349b -r 3c91aabcd9ac sys/dev/pci/machfb.c
--- a/sys/dev/pci/machfb.c      Fri Aug 07 18:14:21 2020 +0000
+++ b/sys/dev/pci/machfb.c      Fri Aug 07 18:26:33 2020 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: machfb.c,v 1.100 2020/07/30 21:29:20 macallan Exp $    */
+/*     $NetBSD: machfb.c,v 1.101 2020/08/07 18:26:33 jdc Exp $ */
 
 /*
  * Copyright (c) 2002 Bang Jun-Young
@@ -34,7 +34,7 @@
 
 #include <sys/cdefs.h>
 __KERNEL_RCSID(0,
-       "$NetBSD: machfb.c,v 1.100 2020/07/30 21:29:20 macallan Exp $");
+       "$NetBSD: machfb.c,v 1.101 2020/08/07 18:26:33 jdc Exp $");
 
 #include <sys/param.h>
 #include <sys/systm.h>
@@ -228,8 +228,9 @@
 static void    mach64_init(struct mach64_softc *);
 static int     mach64_get_memsize(struct mach64_softc *);
 static int     mach64_get_max_ramdac(struct mach64_softc *);
+static int     mach64_ref_freq(void);
 
-#if 0
+#ifdef MACHFB_DEBUG
 static void    mach64_get_mode(struct mach64_softc *, struct videomode *);
 #endif
 
@@ -505,6 +506,9 @@
 
        printf("%s: %d KB ROM at 0x%08x\n", device_xname(sc->sc_dev),
            (int)sc->sc_rom.vb_size >> 10, (uint32_t)sc->sc_rom.vb_base);
+#ifdef MACHFB_DEBUG
+       mach64_get_mode(sc, NULL);
+#endif
 
        prop_dictionary_get_uint32(device_properties(self), "width", &width);
        prop_dictionary_get_uint32(device_properties(self), "height", &height);
@@ -556,23 +560,13 @@
        else
                sc->memtype = regr(sc, CONFIG_STAT0) & 0x07;
 
-       /*
-        * XXX is there any way to calculate reference frequency from
-        * known values?
-        */
-       if ((mach64_chip_id == PCI_PRODUCT_ATI_RAGE_XL_PCI) ||
-           ((mach64_chip_id >= PCI_PRODUCT_ATI_RAGE_LT_PRO_PCI) &&
-           (mach64_chip_id <= PCI_PRODUCT_ATI_RAGE_LT_PRO))) {
-               aprint_normal_dev(sc->sc_dev, "ref_freq=29.498MHz\n");
-               sc->ref_freq = 29498;
-       } else
-               sc->ref_freq = 14318;
+       sc->ref_freq = mach64_ref_freq();
 
        reg = regr(sc, CLOCK_CNTL);
-       aprint_debug("CLOCK_CNTL: %08x\n", reg);
        sc->sc_clock = reg & 3;
-       aprint_debug("using clock %d\n", sc->sc_clock);
+       DPRINTF("using clock %d\n", sc->sc_clock);
 
+       DPRINTF("ref_freq: %d\n", sc->ref_freq);
        sc->ref_div = regrb_pll(sc, PLL_REF_DIV);
        DPRINTF("ref_div: %d\n", sc->ref_div);
        sc->mclk_fb_div = regrb_pll(sc, MCLK_FB_DIV);
@@ -636,8 +630,9 @@
                                    sizeof(struct videomode));
                                sc->sc_setmode = 1;
                        } else {
-                               aprint_error_dev(sc->sc_dev,
-                                   "unable to use preferred mode\n");
+                               aprint_normal_dev(sc->sc_dev,
+                                   "unable to use EDID preferred mode "
+                                   "(%d x %d)\n", m->hdisplay, m->vdisplay);
                        }
                }
                /*
@@ -905,32 +900,118 @@
                return 80000;
 }
 
-#if 0
+static int
+mach64_ref_freq(void)
+{
+       /*
+        * There doesn't seem to be any way to calculate the reference
+        * frequency from known values
+        */
+       if ((mach64_chip_id == PCI_PRODUCT_ATI_RAGE_XL_PCI) ||
+           ((mach64_chip_id >= PCI_PRODUCT_ATI_RAGE_LT_PRO_PCI) &&
+           (mach64_chip_id <= PCI_PRODUCT_ATI_RAGE_L_MOB_M1_PCI)))
+               return 29498;
+       else
+               return 14318;
+}
+
+#ifdef MACHFB_DEBUG
 static void
 mach64_get_mode(struct mach64_softc *sc, struct videomode *mode)
 {
-       struct mach64_crtcregs crtc;
+       int htotal, hdisplay, hsync_start, hsync_end;
+       int vtotal, vdisplay, vsync_start, vsync_end;
+       int gen_ctl, clk_ctl, clock;
+       int ref_freq, ref_div, mclk_fb_div, vclk_post_div, vclk_fb_div;
+       int nhsync, nvsync;
+       int post_div, dot_clock, vrefresh, vrefresh2;
+
+       hdisplay = regr(sc, CRTC_H_TOTAL_DISP);
+       hsync_end = regr(sc, CRTC_H_SYNC_STRT_WID);
+       vdisplay = regr(sc, CRTC_V_TOTAL_DISP);
+       vsync_end = regr(sc, CRTC_V_SYNC_STRT_WID);
+       gen_ctl = regr(sc, CRTC_GEN_CNTL);
+       clk_ctl = regr(sc, CLOCK_CNTL);
+       clock = clk_ctl & 3;
+       ref_div = regrb_pll(sc, PLL_REF_DIV);
+       mclk_fb_div = regrb_pll(sc, MCLK_FB_DIV);
+       vclk_post_div = regrb_pll(sc, VCLK_POST_DIV);
+       vclk_fb_div = regrb_pll(sc, VCLK0_FB_DIV + clock);
+       ref_freq = mach64_ref_freq();
 
-       crtc.h_total_disp = regr(sc, CRTC_H_TOTAL_DISP);
-       crtc.h_sync_strt_wid = regr(sc, CRTC_H_SYNC_STRT_WID);
-       crtc.v_total_disp = regr(sc, CRTC_V_TOTAL_DISP);
-       crtc.v_sync_strt_wid = regr(sc, CRTC_V_SYNC_STRT_WID);
+       aprint_normal_dev(sc->sc_dev, "CRTC registers:\n");
+       aprint_normal("\th total: 0x%08x  h sync: 0x%08x\n",
+           hdisplay, hsync_end);
+       aprint_normal("\tv total: 0x%08x  v sync: 0x%08x\n",
+           vdisplay, vsync_end);
+       aprint_normal("\t g cntl: 0x%08x  c cntl: 0x%08x\n",
+           gen_ctl, clk_ctl);
+       aprint_normal("\t rfreq %d  rdiv: %d\n", ref_freq, ref_div);
+       aprint_normal_dev(sc->sc_dev, "PLL registers:\n");
+       aprint_normal("\t m div: 0x%02x  p div: 0x%02x  v%d div: 0x%02x\n",
+           mclk_fb_div, vclk_post_div, clock, vclk_fb_div);
+
+       htotal = ((hdisplay & 0x01ff) + 1) << 3;
+       hdisplay = (((hdisplay & 0x1ff0000) >> 16) + 1) << 3;
+       if (hsync_end & CRTC_HSYNC_NEG)
+               nhsync = 1;
+       else
+               nhsync = 0;
+       hsync_start = (((hsync_end & 0xff) + 1) << 3) +
+           ((hsync_end & 0x700) >> 8);
+       hsync_end = (((hsync_end & 0x1f0000) >> 16) << 3) + hsync_start;
 
-       mode->htotal = ((crtc.h_total_disp & 0xffff) + 1) << 3;
-       mode->hdisplay = ((crtc.h_total_disp >> 16) + 1) << 3;
-       mode->hsync_start = ((crtc.h_sync_strt_wid & 0xffff) + 1) << 3;
-       mode->hsync_end = ((crtc.h_sync_strt_wid >> 16) << 3) +
-           mode->hsync_start;
-       mode->vtotal = (crtc.v_total_disp & 0xffff) + 1;
-       mode->vdisplay = (crtc.v_total_disp >> 16) + 1;
-       mode->vsync_start = (crtc.v_sync_strt_wid & 0xffff) + 1;
-       mode->vsync_end = (crtc.v_sync_strt_wid >> 16) + mode->vsync_start;
+       vtotal = (vdisplay & 0x07ff) + 1;
+       vdisplay = ((vdisplay & 0x7ff0000) >> 16) + 1;
+       if (vsync_end & CRTC_VSYNC_NEG)
+               nvsync = 1;
+       else
+               nvsync = 0;
+       vsync_start = (vsync_end & 0x07ff) + 1;
+       vsync_end = ((vsync_end & 0x1f0000) >> 16) + vsync_start;
 
-#ifdef MACHFB_DEBUG
-       printf("mach64_get_mode: %d %d %d %d %d %d %d %d\n",
-           mode->hdisplay, mode->hsync_start, mode->hsync_end, mode->htotal,
-           mode->vdisplay, mode->vsync_start, mode->vsync_end, mode->vtotal);
-#endif
+       switch ((vclk_post_div >> (clock * 2)) & 3) {
+               case 3:
+                       post_div = 8;
+                       break;
+               case 2:
+                       post_div = 4;
+                       break;
+               case 1:
+                       post_div = 2;
+                       break;
+               default:
+                       post_div = 1;
+                       break;
+       }
+       dot_clock = (2 * ref_freq * vclk_fb_div) / (ref_div * post_div);
+       vrefresh = (dot_clock * 1000) / (htotal * vtotal);
+       vrefresh2 = ((dot_clock * 1000) - (vrefresh * htotal * vtotal)) * 100 /
+           (htotal * vtotal);
+
+       aprint_normal_dev(sc->sc_dev, "Video mode:\n");
+       aprint_normal("\t%d" "x%d @ %d.%02dHz "
+           "(%d %d %d %d %d %d %d %cH %cV)\n",
+           hdisplay, vdisplay, vrefresh, vrefresh2, dot_clock,
+           hsync_start, hsync_end, htotal, vsync_start, vsync_end, vtotal,
+           nhsync == 1 ? '-' : '+', nvsync == 1 ? '-' : '+');
+
+       if (mode != NULL) {
+               mode->dot_clock = dot_clock;
+               mode->htotal = htotal;
+               mode->hdisplay = hdisplay;
+               mode->hsync_start = hsync_start;
+               mode->hsync_end = hsync_end;
+               mode->vtotal = vtotal;
+               mode->vdisplay = vdisplay;
+               mode->vsync_start = vsync_start;
+               mode->vsync_end = vsync_end;
+               mode->flags = 0;
+               if (nhsync)
+                       mode->flags |= VID_NHSYNC;
+               if (nvsync)
+                       mode->flags |= VID_NVSYNC;
+       }
 }
 #endif
 
@@ -947,7 +1028,7 @@
            ((mode->htotal >> 3) - 1);
        crtc->h_sync_strt_wid =
            (((mode->hsync_end - mode->hsync_start) >> 3) << 16) |
-           ((mode->hsync_start >> 3) - 1);
+           ((mode->hsync_start >> 3) - 1) | ((mode->hsync_start & 7) << 8);
 
        crtc->v_total_disp = ((mode->vdisplay - 1) << 16) |
            (mode->vtotal - 1);
@@ -991,6 +1072,10 @@
        if (sc->has_dsp)
                mach64_set_dsp(sc);
 
+       DPRINTF("\th total: 0x%08x  h sync: 0x%08x\n",
+           crtc->h_total_disp, crtc->h_sync_strt_wid);
+       DPRINTF("\tv total: 0x%08x  v sync: 0x%08x\n",
+           crtc->v_total_disp, crtc->v_sync_strt_wid);
        regw(sc, CRTC_H_TOTAL_DISP, crtc->h_total_disp);
        regw(sc, CRTC_H_SYNC_STRT_WID, crtc->h_sync_strt_wid);
        regw(sc, CRTC_V_TOTAL_DISP, crtc->v_total_disp);
@@ -1170,7 +1255,7 @@
                       (sc->vclk_freq * sc->bits_per_pixel);
 
        DPRINTF("xclks_per_qw %d %d\n", xclks_per_qw >> 7, xclks_per_qw_m);
-               DPRINTF("mem %dkHz v %dkHz\n", sc->mem_freq, sc->vclk_freq);
+       DPRINTF("mem %dkHz v %dkHz\n", sc->mem_freq, sc->vclk_freq);
 
        y = (xclks_per_qw * fifo_depth) >> 11;
                       
@@ -1224,15 +1309,13 @@
        dsp_on = fifo_on >> dsp_precision;
        dsp_off = fifo_off >> dsp_precision;
 
-#ifdef MACHFB_DEBUG
-       printf("dsp_xclks_per_qw = %d, dsp_on = %d, dsp_off = %d,\n"
+       DPRINTF("dsp_xclks_per_qw = %d, dsp_on = %d, dsp_off = %d,\n"
            "dsp_precision = %d, dsp_loop_latency = %d,\n"
            "mclk_fb_div = %d, vclk_fb_div = %d,\n"
            "mclk_post_div = %d, vclk_post_div = %d\n",
            dsp_xclks_per_qw, dsp_on, dsp_off, dsp_precision, dsp_loop_latency,
            sc->mclk_fb_div, sc->vclk_fb_div,
            sc->mclk_post_div, sc->vclk_post_div);
-#endif
        DPRINTF("DSP_ON_OFF %08x\n", regr(sc, DSP_ON_OFF));
        DPRINTF("DSP_CONFIG %08x\n", regr(sc, DSP_CONFIG));
        regw(sc, DSP_ON_OFF, ((dsp_on << 16) & DSP_ON) | (dsp_off & DSP_OFF));



Home | Main Index | Thread Index | Old Index