18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * 48c2ecf20Sopenharmony_ci * Hardware accelerated Matrox Millennium I, II, Mystique, G100, G200, G400 and G450. 58c2ecf20Sopenharmony_ci * 68c2ecf20Sopenharmony_ci * (c) 1998-2002 Petr Vandrovec <vandrove@vc.cvut.cz> 78c2ecf20Sopenharmony_ci * 88c2ecf20Sopenharmony_ci * Portions Copyright (c) 2001 Matrox Graphics Inc. 98c2ecf20Sopenharmony_ci * 108c2ecf20Sopenharmony_ci * Version: 1.65 2002/08/14 118c2ecf20Sopenharmony_ci * 128c2ecf20Sopenharmony_ci * See matroxfb_base.c for contributors. 138c2ecf20Sopenharmony_ci * 148c2ecf20Sopenharmony_ci */ 158c2ecf20Sopenharmony_ci 168c2ecf20Sopenharmony_ci 178c2ecf20Sopenharmony_ci#include "matroxfb_DAC1064.h" 188c2ecf20Sopenharmony_ci#include "matroxfb_misc.h" 198c2ecf20Sopenharmony_ci#include "matroxfb_accel.h" 208c2ecf20Sopenharmony_ci#include "g450_pll.h" 218c2ecf20Sopenharmony_ci#include <linux/matroxfb.h> 228c2ecf20Sopenharmony_ci 238c2ecf20Sopenharmony_ci#ifdef NEED_DAC1064 248c2ecf20Sopenharmony_ci#define outDAC1064 matroxfb_DAC_out 258c2ecf20Sopenharmony_ci#define inDAC1064 matroxfb_DAC_in 268c2ecf20Sopenharmony_ci 278c2ecf20Sopenharmony_ci#define DAC1064_OPT_SCLK_PCI 0x00 288c2ecf20Sopenharmony_ci#define DAC1064_OPT_SCLK_PLL 0x01 298c2ecf20Sopenharmony_ci#define DAC1064_OPT_SCLK_EXT 0x02 308c2ecf20Sopenharmony_ci#define DAC1064_OPT_SCLK_MASK 0x03 318c2ecf20Sopenharmony_ci#define DAC1064_OPT_GDIV1 0x04 /* maybe it is GDIV2 on G100 ?! */ 328c2ecf20Sopenharmony_ci#define DAC1064_OPT_GDIV3 0x00 338c2ecf20Sopenharmony_ci#define DAC1064_OPT_MDIV1 0x08 348c2ecf20Sopenharmony_ci#define DAC1064_OPT_MDIV2 0x00 358c2ecf20Sopenharmony_ci#define DAC1064_OPT_RESERVED 0x10 368c2ecf20Sopenharmony_ci 378c2ecf20Sopenharmony_cistatic void DAC1064_calcclock(const struct matrox_fb_info *minfo, 388c2ecf20Sopenharmony_ci unsigned int freq, unsigned int fmax, 398c2ecf20Sopenharmony_ci unsigned int *in, unsigned int *feed, 408c2ecf20Sopenharmony_ci unsigned int *post) 418c2ecf20Sopenharmony_ci{ 428c2ecf20Sopenharmony_ci unsigned int fvco; 438c2ecf20Sopenharmony_ci unsigned int p; 448c2ecf20Sopenharmony_ci 458c2ecf20Sopenharmony_ci DBG(__func__) 468c2ecf20Sopenharmony_ci 478c2ecf20Sopenharmony_ci /* only for devices older than G450 */ 488c2ecf20Sopenharmony_ci 498c2ecf20Sopenharmony_ci fvco = PLL_calcclock(minfo, freq, fmax, in, feed, &p); 508c2ecf20Sopenharmony_ci 518c2ecf20Sopenharmony_ci p = (1 << p) - 1; 528c2ecf20Sopenharmony_ci if (fvco <= 100000) 538c2ecf20Sopenharmony_ci ; 548c2ecf20Sopenharmony_ci else if (fvco <= 140000) 558c2ecf20Sopenharmony_ci p |= 0x08; 568c2ecf20Sopenharmony_ci else if (fvco <= 180000) 578c2ecf20Sopenharmony_ci p |= 0x10; 588c2ecf20Sopenharmony_ci else 598c2ecf20Sopenharmony_ci p |= 0x18; 608c2ecf20Sopenharmony_ci *post = p; 618c2ecf20Sopenharmony_ci} 628c2ecf20Sopenharmony_ci 638c2ecf20Sopenharmony_ci/* they must be in POS order */ 648c2ecf20Sopenharmony_cistatic const unsigned char MGA1064_DAC_regs[] = { 658c2ecf20Sopenharmony_ci M1064_XCURADDL, M1064_XCURADDH, M1064_XCURCTRL, 668c2ecf20Sopenharmony_ci M1064_XCURCOL0RED, M1064_XCURCOL0GREEN, M1064_XCURCOL0BLUE, 678c2ecf20Sopenharmony_ci M1064_XCURCOL1RED, M1064_XCURCOL1GREEN, M1064_XCURCOL1BLUE, 688c2ecf20Sopenharmony_ci M1064_XCURCOL2RED, M1064_XCURCOL2GREEN, M1064_XCURCOL2BLUE, 698c2ecf20Sopenharmony_ci DAC1064_XVREFCTRL, M1064_XMULCTRL, M1064_XPIXCLKCTRL, M1064_XGENCTRL, 708c2ecf20Sopenharmony_ci M1064_XMISCCTRL, 718c2ecf20Sopenharmony_ci M1064_XGENIOCTRL, M1064_XGENIODATA, M1064_XZOOMCTRL, M1064_XSENSETEST, 728c2ecf20Sopenharmony_ci M1064_XCRCBITSEL, 738c2ecf20Sopenharmony_ci M1064_XCOLKEYMASKL, M1064_XCOLKEYMASKH, M1064_XCOLKEYL, M1064_XCOLKEYH }; 748c2ecf20Sopenharmony_ci 758c2ecf20Sopenharmony_cistatic const unsigned char MGA1064_DAC[] = { 768c2ecf20Sopenharmony_ci 0x00, 0x00, M1064_XCURCTRL_DIS, 778c2ecf20Sopenharmony_ci 0x00, 0x00, 0x00, /* black */ 788c2ecf20Sopenharmony_ci 0xFF, 0xFF, 0xFF, /* white */ 798c2ecf20Sopenharmony_ci 0xFF, 0x00, 0x00, /* red */ 808c2ecf20Sopenharmony_ci 0x00, 0, 818c2ecf20Sopenharmony_ci M1064_XPIXCLKCTRL_PLL_UP | M1064_XPIXCLKCTRL_EN | M1064_XPIXCLKCTRL_SRC_PLL, 828c2ecf20Sopenharmony_ci M1064_XGENCTRL_VS_0 | M1064_XGENCTRL_ALPHA_DIS | M1064_XGENCTRL_BLACK_0IRE | M1064_XGENCTRL_NO_SYNC_ON_GREEN, 838c2ecf20Sopenharmony_ci M1064_XMISCCTRL_DAC_8BIT, 848c2ecf20Sopenharmony_ci 0x00, 0x00, M1064_XZOOMCTRL_1, M1064_XSENSETEST_BCOMP | M1064_XSENSETEST_GCOMP | M1064_XSENSETEST_RCOMP | M1064_XSENSETEST_PDOWN, 858c2ecf20Sopenharmony_ci 0x00, 868c2ecf20Sopenharmony_ci 0x00, 0x00, 0xFF, 0xFF}; 878c2ecf20Sopenharmony_ci 888c2ecf20Sopenharmony_cistatic void DAC1064_setpclk(struct matrox_fb_info *minfo, unsigned long fout) 898c2ecf20Sopenharmony_ci{ 908c2ecf20Sopenharmony_ci unsigned int m, n, p; 918c2ecf20Sopenharmony_ci 928c2ecf20Sopenharmony_ci DBG(__func__) 938c2ecf20Sopenharmony_ci 948c2ecf20Sopenharmony_ci DAC1064_calcclock(minfo, fout, minfo->max_pixel_clock, &m, &n, &p); 958c2ecf20Sopenharmony_ci minfo->hw.DACclk[0] = m; 968c2ecf20Sopenharmony_ci minfo->hw.DACclk[1] = n; 978c2ecf20Sopenharmony_ci minfo->hw.DACclk[2] = p; 988c2ecf20Sopenharmony_ci} 998c2ecf20Sopenharmony_ci 1008c2ecf20Sopenharmony_cistatic void DAC1064_setmclk(struct matrox_fb_info *minfo, int oscinfo, 1018c2ecf20Sopenharmony_ci unsigned long fmem) 1028c2ecf20Sopenharmony_ci{ 1038c2ecf20Sopenharmony_ci u_int32_t mx; 1048c2ecf20Sopenharmony_ci struct matrox_hw_state *hw = &minfo->hw; 1058c2ecf20Sopenharmony_ci 1068c2ecf20Sopenharmony_ci DBG(__func__) 1078c2ecf20Sopenharmony_ci 1088c2ecf20Sopenharmony_ci if (minfo->devflags.noinit) { 1098c2ecf20Sopenharmony_ci /* read MCLK and give up... */ 1108c2ecf20Sopenharmony_ci hw->DACclk[3] = inDAC1064(minfo, DAC1064_XSYSPLLM); 1118c2ecf20Sopenharmony_ci hw->DACclk[4] = inDAC1064(minfo, DAC1064_XSYSPLLN); 1128c2ecf20Sopenharmony_ci hw->DACclk[5] = inDAC1064(minfo, DAC1064_XSYSPLLP); 1138c2ecf20Sopenharmony_ci return; 1148c2ecf20Sopenharmony_ci } 1158c2ecf20Sopenharmony_ci mx = hw->MXoptionReg | 0x00000004; 1168c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION_REG, mx); 1178c2ecf20Sopenharmony_ci mx &= ~0x000000BB; 1188c2ecf20Sopenharmony_ci if (oscinfo & DAC1064_OPT_GDIV1) 1198c2ecf20Sopenharmony_ci mx |= 0x00000008; 1208c2ecf20Sopenharmony_ci if (oscinfo & DAC1064_OPT_MDIV1) 1218c2ecf20Sopenharmony_ci mx |= 0x00000010; 1228c2ecf20Sopenharmony_ci if (oscinfo & DAC1064_OPT_RESERVED) 1238c2ecf20Sopenharmony_ci mx |= 0x00000080; 1248c2ecf20Sopenharmony_ci if ((oscinfo & DAC1064_OPT_SCLK_MASK) == DAC1064_OPT_SCLK_PLL) { 1258c2ecf20Sopenharmony_ci /* select PCI clock until we have setup oscilator... */ 1268c2ecf20Sopenharmony_ci int clk; 1278c2ecf20Sopenharmony_ci unsigned int m, n, p; 1288c2ecf20Sopenharmony_ci 1298c2ecf20Sopenharmony_ci /* powerup system PLL, select PCI clock */ 1308c2ecf20Sopenharmony_ci mx |= 0x00000020; 1318c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION_REG, mx); 1328c2ecf20Sopenharmony_ci mx &= ~0x00000004; 1338c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION_REG, mx); 1348c2ecf20Sopenharmony_ci 1358c2ecf20Sopenharmony_ci /* !!! you must not access device if MCLK is not running !!! 1368c2ecf20Sopenharmony_ci Doing so cause immediate PCI lockup :-( Maybe they should 1378c2ecf20Sopenharmony_ci generate ABORT or I/O (parity...) error and Linux should 1388c2ecf20Sopenharmony_ci recover from this... (kill driver/process). But world is not 1398c2ecf20Sopenharmony_ci perfect... */ 1408c2ecf20Sopenharmony_ci /* (bit 2 of PCI_OPTION_REG must be 0... and bits 0,1 must not 1418c2ecf20Sopenharmony_ci select PLL... because of PLL can be stopped at this time) */ 1428c2ecf20Sopenharmony_ci DAC1064_calcclock(minfo, fmem, minfo->max_pixel_clock, &m, &n, &p); 1438c2ecf20Sopenharmony_ci outDAC1064(minfo, DAC1064_XSYSPLLM, hw->DACclk[3] = m); 1448c2ecf20Sopenharmony_ci outDAC1064(minfo, DAC1064_XSYSPLLN, hw->DACclk[4] = n); 1458c2ecf20Sopenharmony_ci outDAC1064(minfo, DAC1064_XSYSPLLP, hw->DACclk[5] = p); 1468c2ecf20Sopenharmony_ci for (clk = 65536; clk; --clk) { 1478c2ecf20Sopenharmony_ci if (inDAC1064(minfo, DAC1064_XSYSPLLSTAT) & 0x40) 1488c2ecf20Sopenharmony_ci break; 1498c2ecf20Sopenharmony_ci } 1508c2ecf20Sopenharmony_ci if (!clk) 1518c2ecf20Sopenharmony_ci printk(KERN_ERR "matroxfb: aiee, SYSPLL not locked\n"); 1528c2ecf20Sopenharmony_ci /* select PLL */ 1538c2ecf20Sopenharmony_ci mx |= 0x00000005; 1548c2ecf20Sopenharmony_ci } else { 1558c2ecf20Sopenharmony_ci /* select specified system clock source */ 1568c2ecf20Sopenharmony_ci mx |= oscinfo & DAC1064_OPT_SCLK_MASK; 1578c2ecf20Sopenharmony_ci } 1588c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION_REG, mx); 1598c2ecf20Sopenharmony_ci mx &= ~0x00000004; 1608c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION_REG, mx); 1618c2ecf20Sopenharmony_ci hw->MXoptionReg = mx; 1628c2ecf20Sopenharmony_ci} 1638c2ecf20Sopenharmony_ci 1648c2ecf20Sopenharmony_ci#ifdef CONFIG_FB_MATROX_G 1658c2ecf20Sopenharmony_cistatic void g450_set_plls(struct matrox_fb_info *minfo) 1668c2ecf20Sopenharmony_ci{ 1678c2ecf20Sopenharmony_ci u_int32_t c2_ctl; 1688c2ecf20Sopenharmony_ci unsigned int pxc; 1698c2ecf20Sopenharmony_ci struct matrox_hw_state *hw = &minfo->hw; 1708c2ecf20Sopenharmony_ci int pixelmnp; 1718c2ecf20Sopenharmony_ci int videomnp; 1728c2ecf20Sopenharmony_ci 1738c2ecf20Sopenharmony_ci c2_ctl = hw->crtc2.ctl & ~0x4007; /* Clear PLL + enable for CRTC2 */ 1748c2ecf20Sopenharmony_ci c2_ctl |= 0x0001; /* Enable CRTC2 */ 1758c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XPWRCTRL] &= ~0x02; /* Stop VIDEO PLL */ 1768c2ecf20Sopenharmony_ci pixelmnp = minfo->crtc1.mnp; 1778c2ecf20Sopenharmony_ci videomnp = minfo->crtc2.mnp; 1788c2ecf20Sopenharmony_ci if (videomnp < 0) { 1798c2ecf20Sopenharmony_ci c2_ctl &= ~0x0001; /* Disable CRTC2 */ 1808c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XPWRCTRL] &= ~0x10; /* Powerdown CRTC2 */ 1818c2ecf20Sopenharmony_ci } else if (minfo->crtc2.pixclock == minfo->features.pll.ref_freq) { 1828c2ecf20Sopenharmony_ci c2_ctl |= 0x4002; /* Use reference directly */ 1838c2ecf20Sopenharmony_ci } else if (videomnp == pixelmnp) { 1848c2ecf20Sopenharmony_ci c2_ctl |= 0x0004; /* Use pixel PLL */ 1858c2ecf20Sopenharmony_ci } else { 1868c2ecf20Sopenharmony_ci if (0 == ((videomnp ^ pixelmnp) & 0xFFFFFF00)) { 1878c2ecf20Sopenharmony_ci /* PIXEL and VIDEO PLL must not use same frequency. We modify N 1888c2ecf20Sopenharmony_ci of PIXEL PLL in such case because of VIDEO PLL may be source 1898c2ecf20Sopenharmony_ci of TVO clocks, and chroma subcarrier is derived from its 1908c2ecf20Sopenharmony_ci pixel clocks */ 1918c2ecf20Sopenharmony_ci pixelmnp += 0x000100; 1928c2ecf20Sopenharmony_ci } 1938c2ecf20Sopenharmony_ci c2_ctl |= 0x0006; /* Use video PLL */ 1948c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XPWRCTRL] |= 0x02; 1958c2ecf20Sopenharmony_ci 1968c2ecf20Sopenharmony_ci outDAC1064(minfo, M1064_XPWRCTRL, hw->DACreg[POS1064_XPWRCTRL]); 1978c2ecf20Sopenharmony_ci matroxfb_g450_setpll_cond(minfo, videomnp, M_VIDEO_PLL); 1988c2ecf20Sopenharmony_ci } 1998c2ecf20Sopenharmony_ci 2008c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XPIXCLKCTRL] &= ~M1064_XPIXCLKCTRL_PLL_UP; 2018c2ecf20Sopenharmony_ci if (pixelmnp >= 0) { 2028c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XPIXCLKCTRL] |= M1064_XPIXCLKCTRL_PLL_UP; 2038c2ecf20Sopenharmony_ci 2048c2ecf20Sopenharmony_ci outDAC1064(minfo, M1064_XPIXCLKCTRL, hw->DACreg[POS1064_XPIXCLKCTRL]); 2058c2ecf20Sopenharmony_ci matroxfb_g450_setpll_cond(minfo, pixelmnp, M_PIXEL_PLL_C); 2068c2ecf20Sopenharmony_ci } 2078c2ecf20Sopenharmony_ci if (c2_ctl != hw->crtc2.ctl) { 2088c2ecf20Sopenharmony_ci hw->crtc2.ctl = c2_ctl; 2098c2ecf20Sopenharmony_ci mga_outl(0x3C10, c2_ctl); 2108c2ecf20Sopenharmony_ci } 2118c2ecf20Sopenharmony_ci 2128c2ecf20Sopenharmony_ci pxc = minfo->crtc1.pixclock; 2138c2ecf20Sopenharmony_ci if (pxc == 0 || minfo->outputs[2].src == MATROXFB_SRC_CRTC2) { 2148c2ecf20Sopenharmony_ci pxc = minfo->crtc2.pixclock; 2158c2ecf20Sopenharmony_ci } 2168c2ecf20Sopenharmony_ci if (minfo->chip == MGA_G550) { 2178c2ecf20Sopenharmony_ci if (pxc < 45000) { 2188c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XPANMODE] = 0x00; /* 0-50 */ 2198c2ecf20Sopenharmony_ci } else if (pxc < 55000) { 2208c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XPANMODE] = 0x08; /* 34-62 */ 2218c2ecf20Sopenharmony_ci } else if (pxc < 70000) { 2228c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XPANMODE] = 0x10; /* 42-78 */ 2238c2ecf20Sopenharmony_ci } else if (pxc < 85000) { 2248c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XPANMODE] = 0x18; /* 62-92 */ 2258c2ecf20Sopenharmony_ci } else if (pxc < 100000) { 2268c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XPANMODE] = 0x20; /* 74-108 */ 2278c2ecf20Sopenharmony_ci } else if (pxc < 115000) { 2288c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XPANMODE] = 0x28; /* 94-122 */ 2298c2ecf20Sopenharmony_ci } else if (pxc < 125000) { 2308c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XPANMODE] = 0x30; /* 108-132 */ 2318c2ecf20Sopenharmony_ci } else { 2328c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XPANMODE] = 0x38; /* 120-168 */ 2338c2ecf20Sopenharmony_ci } 2348c2ecf20Sopenharmony_ci } else { 2358c2ecf20Sopenharmony_ci /* G450 */ 2368c2ecf20Sopenharmony_ci if (pxc < 45000) { 2378c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XPANMODE] = 0x00; /* 0-54 */ 2388c2ecf20Sopenharmony_ci } else if (pxc < 65000) { 2398c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XPANMODE] = 0x08; /* 38-70 */ 2408c2ecf20Sopenharmony_ci } else if (pxc < 85000) { 2418c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XPANMODE] = 0x10; /* 56-96 */ 2428c2ecf20Sopenharmony_ci } else if (pxc < 105000) { 2438c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XPANMODE] = 0x18; /* 80-114 */ 2448c2ecf20Sopenharmony_ci } else if (pxc < 135000) { 2458c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XPANMODE] = 0x20; /* 102-144 */ 2468c2ecf20Sopenharmony_ci } else if (pxc < 160000) { 2478c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XPANMODE] = 0x28; /* 132-166 */ 2488c2ecf20Sopenharmony_ci } else if (pxc < 175000) { 2498c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XPANMODE] = 0x30; /* 154-182 */ 2508c2ecf20Sopenharmony_ci } else { 2518c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XPANMODE] = 0x38; /* 170-204 */ 2528c2ecf20Sopenharmony_ci } 2538c2ecf20Sopenharmony_ci } 2548c2ecf20Sopenharmony_ci} 2558c2ecf20Sopenharmony_ci#endif 2568c2ecf20Sopenharmony_ci 2578c2ecf20Sopenharmony_civoid DAC1064_global_init(struct matrox_fb_info *minfo) 2588c2ecf20Sopenharmony_ci{ 2598c2ecf20Sopenharmony_ci struct matrox_hw_state *hw = &minfo->hw; 2608c2ecf20Sopenharmony_ci 2618c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XMISCCTRL] &= M1064_XMISCCTRL_DAC_WIDTHMASK; 2628c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XMISCCTRL] |= M1064_XMISCCTRL_LUT_EN; 2638c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XPIXCLKCTRL] = M1064_XPIXCLKCTRL_PLL_UP | M1064_XPIXCLKCTRL_EN | M1064_XPIXCLKCTRL_SRC_PLL; 2648c2ecf20Sopenharmony_ci#ifdef CONFIG_FB_MATROX_G 2658c2ecf20Sopenharmony_ci if (minfo->devflags.g450dac) { 2668c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XPWRCTRL] = 0x1F; /* powerup everything */ 2678c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XOUTPUTCONN] = 0x00; /* disable outputs */ 2688c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XMISCCTRL] |= M1064_XMISCCTRL_DAC_EN; 2698c2ecf20Sopenharmony_ci switch (minfo->outputs[0].src) { 2708c2ecf20Sopenharmony_ci case MATROXFB_SRC_CRTC1: 2718c2ecf20Sopenharmony_ci case MATROXFB_SRC_CRTC2: 2728c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XOUTPUTCONN] |= 0x01; /* enable output; CRTC1/2 selection is in CRTC2 ctl */ 2738c2ecf20Sopenharmony_ci break; 2748c2ecf20Sopenharmony_ci case MATROXFB_SRC_NONE: 2758c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XMISCCTRL] &= ~M1064_XMISCCTRL_DAC_EN; 2768c2ecf20Sopenharmony_ci break; 2778c2ecf20Sopenharmony_ci } 2788c2ecf20Sopenharmony_ci switch (minfo->outputs[1].src) { 2798c2ecf20Sopenharmony_ci case MATROXFB_SRC_CRTC1: 2808c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XOUTPUTCONN] |= 0x04; 2818c2ecf20Sopenharmony_ci break; 2828c2ecf20Sopenharmony_ci case MATROXFB_SRC_CRTC2: 2838c2ecf20Sopenharmony_ci if (minfo->outputs[1].mode == MATROXFB_OUTPUT_MODE_MONITOR) { 2848c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XOUTPUTCONN] |= 0x08; 2858c2ecf20Sopenharmony_ci } else { 2868c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XOUTPUTCONN] |= 0x0C; 2878c2ecf20Sopenharmony_ci } 2888c2ecf20Sopenharmony_ci break; 2898c2ecf20Sopenharmony_ci case MATROXFB_SRC_NONE: 2908c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XPWRCTRL] &= ~0x01; /* Poweroff DAC2 */ 2918c2ecf20Sopenharmony_ci break; 2928c2ecf20Sopenharmony_ci } 2938c2ecf20Sopenharmony_ci switch (minfo->outputs[2].src) { 2948c2ecf20Sopenharmony_ci case MATROXFB_SRC_CRTC1: 2958c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XOUTPUTCONN] |= 0x20; 2968c2ecf20Sopenharmony_ci break; 2978c2ecf20Sopenharmony_ci case MATROXFB_SRC_CRTC2: 2988c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XOUTPUTCONN] |= 0x40; 2998c2ecf20Sopenharmony_ci break; 3008c2ecf20Sopenharmony_ci case MATROXFB_SRC_NONE: 3018c2ecf20Sopenharmony_ci#if 0 3028c2ecf20Sopenharmony_ci /* HELP! If we boot without DFP connected to DVI, we can 3038c2ecf20Sopenharmony_ci poweroff TMDS. But if we boot with DFP connected, 3048c2ecf20Sopenharmony_ci TMDS generated clocks are used instead of ALL pixclocks 3058c2ecf20Sopenharmony_ci available... If someone knows which register 3068c2ecf20Sopenharmony_ci handles it, please reveal this secret to me... */ 3078c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XPWRCTRL] &= ~0x04; /* Poweroff TMDS */ 3088c2ecf20Sopenharmony_ci#endif 3098c2ecf20Sopenharmony_ci break; 3108c2ecf20Sopenharmony_ci } 3118c2ecf20Sopenharmony_ci /* Now set timming related variables... */ 3128c2ecf20Sopenharmony_ci g450_set_plls(minfo); 3138c2ecf20Sopenharmony_ci } else 3148c2ecf20Sopenharmony_ci#endif 3158c2ecf20Sopenharmony_ci { 3168c2ecf20Sopenharmony_ci if (minfo->outputs[1].src == MATROXFB_SRC_CRTC1) { 3178c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XPIXCLKCTRL] = M1064_XPIXCLKCTRL_PLL_UP | M1064_XPIXCLKCTRL_EN | M1064_XPIXCLKCTRL_SRC_EXT; 3188c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XMISCCTRL] |= GX00_XMISCCTRL_MFC_MAFC | G400_XMISCCTRL_VDO_MAFC12; 3198c2ecf20Sopenharmony_ci } else if (minfo->outputs[1].src == MATROXFB_SRC_CRTC2) { 3208c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XMISCCTRL] |= GX00_XMISCCTRL_MFC_MAFC | G400_XMISCCTRL_VDO_C2_MAFC12; 3218c2ecf20Sopenharmony_ci } else if (minfo->outputs[2].src == MATROXFB_SRC_CRTC1) 3228c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XMISCCTRL] |= GX00_XMISCCTRL_MFC_PANELLINK | G400_XMISCCTRL_VDO_MAFC12; 3238c2ecf20Sopenharmony_ci else 3248c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XMISCCTRL] |= GX00_XMISCCTRL_MFC_DIS; 3258c2ecf20Sopenharmony_ci 3268c2ecf20Sopenharmony_ci if (minfo->outputs[0].src != MATROXFB_SRC_NONE) 3278c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XMISCCTRL] |= M1064_XMISCCTRL_DAC_EN; 3288c2ecf20Sopenharmony_ci } 3298c2ecf20Sopenharmony_ci} 3308c2ecf20Sopenharmony_ci 3318c2ecf20Sopenharmony_civoid DAC1064_global_restore(struct matrox_fb_info *minfo) 3328c2ecf20Sopenharmony_ci{ 3338c2ecf20Sopenharmony_ci struct matrox_hw_state *hw = &minfo->hw; 3348c2ecf20Sopenharmony_ci 3358c2ecf20Sopenharmony_ci outDAC1064(minfo, M1064_XPIXCLKCTRL, hw->DACreg[POS1064_XPIXCLKCTRL]); 3368c2ecf20Sopenharmony_ci outDAC1064(minfo, M1064_XMISCCTRL, hw->DACreg[POS1064_XMISCCTRL]); 3378c2ecf20Sopenharmony_ci if (minfo->devflags.accelerator == FB_ACCEL_MATROX_MGAG400) { 3388c2ecf20Sopenharmony_ci outDAC1064(minfo, 0x20, 0x04); 3398c2ecf20Sopenharmony_ci outDAC1064(minfo, 0x1F, minfo->devflags.dfp_type); 3408c2ecf20Sopenharmony_ci if (minfo->devflags.g450dac) { 3418c2ecf20Sopenharmony_ci outDAC1064(minfo, M1064_XSYNCCTRL, 0xCC); 3428c2ecf20Sopenharmony_ci outDAC1064(minfo, M1064_XPWRCTRL, hw->DACreg[POS1064_XPWRCTRL]); 3438c2ecf20Sopenharmony_ci outDAC1064(minfo, M1064_XPANMODE, hw->DACreg[POS1064_XPANMODE]); 3448c2ecf20Sopenharmony_ci outDAC1064(minfo, M1064_XOUTPUTCONN, hw->DACreg[POS1064_XOUTPUTCONN]); 3458c2ecf20Sopenharmony_ci } 3468c2ecf20Sopenharmony_ci } 3478c2ecf20Sopenharmony_ci} 3488c2ecf20Sopenharmony_ci 3498c2ecf20Sopenharmony_cistatic int DAC1064_init_1(struct matrox_fb_info *minfo, struct my_timming *m) 3508c2ecf20Sopenharmony_ci{ 3518c2ecf20Sopenharmony_ci struct matrox_hw_state *hw = &minfo->hw; 3528c2ecf20Sopenharmony_ci 3538c2ecf20Sopenharmony_ci DBG(__func__) 3548c2ecf20Sopenharmony_ci 3558c2ecf20Sopenharmony_ci memcpy(hw->DACreg, MGA1064_DAC, sizeof(MGA1064_DAC_regs)); 3568c2ecf20Sopenharmony_ci switch (minfo->fbcon.var.bits_per_pixel) { 3578c2ecf20Sopenharmony_ci /* case 4: not supported by MGA1064 DAC */ 3588c2ecf20Sopenharmony_ci case 8: 3598c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XMULCTRL] = M1064_XMULCTRL_DEPTH_8BPP | M1064_XMULCTRL_GRAPHICS_PALETIZED; 3608c2ecf20Sopenharmony_ci break; 3618c2ecf20Sopenharmony_ci case 16: 3628c2ecf20Sopenharmony_ci if (minfo->fbcon.var.green.length == 5) 3638c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XMULCTRL] = M1064_XMULCTRL_DEPTH_15BPP_1BPP | M1064_XMULCTRL_GRAPHICS_PALETIZED; 3648c2ecf20Sopenharmony_ci else 3658c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XMULCTRL] = M1064_XMULCTRL_DEPTH_16BPP | M1064_XMULCTRL_GRAPHICS_PALETIZED; 3668c2ecf20Sopenharmony_ci break; 3678c2ecf20Sopenharmony_ci case 24: 3688c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XMULCTRL] = M1064_XMULCTRL_DEPTH_24BPP | M1064_XMULCTRL_GRAPHICS_PALETIZED; 3698c2ecf20Sopenharmony_ci break; 3708c2ecf20Sopenharmony_ci case 32: 3718c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XMULCTRL] = M1064_XMULCTRL_DEPTH_32BPP | M1064_XMULCTRL_GRAPHICS_PALETIZED; 3728c2ecf20Sopenharmony_ci break; 3738c2ecf20Sopenharmony_ci default: 3748c2ecf20Sopenharmony_ci return 1; /* unsupported depth */ 3758c2ecf20Sopenharmony_ci } 3768c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XVREFCTRL] = minfo->features.DAC1064.xvrefctrl; 3778c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XGENCTRL] &= ~M1064_XGENCTRL_SYNC_ON_GREEN_MASK; 3788c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XGENCTRL] |= (m->sync & FB_SYNC_ON_GREEN)?M1064_XGENCTRL_SYNC_ON_GREEN:M1064_XGENCTRL_NO_SYNC_ON_GREEN; 3798c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XCURADDL] = 0; 3808c2ecf20Sopenharmony_ci hw->DACreg[POS1064_XCURADDH] = 0; 3818c2ecf20Sopenharmony_ci 3828c2ecf20Sopenharmony_ci DAC1064_global_init(minfo); 3838c2ecf20Sopenharmony_ci return 0; 3848c2ecf20Sopenharmony_ci} 3858c2ecf20Sopenharmony_ci 3868c2ecf20Sopenharmony_cistatic int DAC1064_init_2(struct matrox_fb_info *minfo, struct my_timming *m) 3878c2ecf20Sopenharmony_ci{ 3888c2ecf20Sopenharmony_ci struct matrox_hw_state *hw = &minfo->hw; 3898c2ecf20Sopenharmony_ci 3908c2ecf20Sopenharmony_ci DBG(__func__) 3918c2ecf20Sopenharmony_ci 3928c2ecf20Sopenharmony_ci if (minfo->fbcon.var.bits_per_pixel > 16) { /* 256 entries */ 3938c2ecf20Sopenharmony_ci int i; 3948c2ecf20Sopenharmony_ci 3958c2ecf20Sopenharmony_ci for (i = 0; i < 256; i++) { 3968c2ecf20Sopenharmony_ci hw->DACpal[i * 3 + 0] = i; 3978c2ecf20Sopenharmony_ci hw->DACpal[i * 3 + 1] = i; 3988c2ecf20Sopenharmony_ci hw->DACpal[i * 3 + 2] = i; 3998c2ecf20Sopenharmony_ci } 4008c2ecf20Sopenharmony_ci } else if (minfo->fbcon.var.bits_per_pixel > 8) { 4018c2ecf20Sopenharmony_ci if (minfo->fbcon.var.green.length == 5) { /* 0..31, 128..159 */ 4028c2ecf20Sopenharmony_ci int i; 4038c2ecf20Sopenharmony_ci 4048c2ecf20Sopenharmony_ci for (i = 0; i < 32; i++) { 4058c2ecf20Sopenharmony_ci /* with p15 == 0 */ 4068c2ecf20Sopenharmony_ci hw->DACpal[i * 3 + 0] = i << 3; 4078c2ecf20Sopenharmony_ci hw->DACpal[i * 3 + 1] = i << 3; 4088c2ecf20Sopenharmony_ci hw->DACpal[i * 3 + 2] = i << 3; 4098c2ecf20Sopenharmony_ci /* with p15 == 1 */ 4108c2ecf20Sopenharmony_ci hw->DACpal[(i + 128) * 3 + 0] = i << 3; 4118c2ecf20Sopenharmony_ci hw->DACpal[(i + 128) * 3 + 1] = i << 3; 4128c2ecf20Sopenharmony_ci hw->DACpal[(i + 128) * 3 + 2] = i << 3; 4138c2ecf20Sopenharmony_ci } 4148c2ecf20Sopenharmony_ci } else { 4158c2ecf20Sopenharmony_ci int i; 4168c2ecf20Sopenharmony_ci 4178c2ecf20Sopenharmony_ci for (i = 0; i < 64; i++) { /* 0..63 */ 4188c2ecf20Sopenharmony_ci hw->DACpal[i * 3 + 0] = i << 3; 4198c2ecf20Sopenharmony_ci hw->DACpal[i * 3 + 1] = i << 2; 4208c2ecf20Sopenharmony_ci hw->DACpal[i * 3 + 2] = i << 3; 4218c2ecf20Sopenharmony_ci } 4228c2ecf20Sopenharmony_ci } 4238c2ecf20Sopenharmony_ci } else { 4248c2ecf20Sopenharmony_ci memset(hw->DACpal, 0, 768); 4258c2ecf20Sopenharmony_ci } 4268c2ecf20Sopenharmony_ci return 0; 4278c2ecf20Sopenharmony_ci} 4288c2ecf20Sopenharmony_ci 4298c2ecf20Sopenharmony_cistatic void DAC1064_restore_1(struct matrox_fb_info *minfo) 4308c2ecf20Sopenharmony_ci{ 4318c2ecf20Sopenharmony_ci struct matrox_hw_state *hw = &minfo->hw; 4328c2ecf20Sopenharmony_ci 4338c2ecf20Sopenharmony_ci CRITFLAGS 4348c2ecf20Sopenharmony_ci 4358c2ecf20Sopenharmony_ci DBG(__func__) 4368c2ecf20Sopenharmony_ci 4378c2ecf20Sopenharmony_ci CRITBEGIN 4388c2ecf20Sopenharmony_ci 4398c2ecf20Sopenharmony_ci if ((inDAC1064(minfo, DAC1064_XSYSPLLM) != hw->DACclk[3]) || 4408c2ecf20Sopenharmony_ci (inDAC1064(minfo, DAC1064_XSYSPLLN) != hw->DACclk[4]) || 4418c2ecf20Sopenharmony_ci (inDAC1064(minfo, DAC1064_XSYSPLLP) != hw->DACclk[5])) { 4428c2ecf20Sopenharmony_ci outDAC1064(minfo, DAC1064_XSYSPLLM, hw->DACclk[3]); 4438c2ecf20Sopenharmony_ci outDAC1064(minfo, DAC1064_XSYSPLLN, hw->DACclk[4]); 4448c2ecf20Sopenharmony_ci outDAC1064(minfo, DAC1064_XSYSPLLP, hw->DACclk[5]); 4458c2ecf20Sopenharmony_ci } 4468c2ecf20Sopenharmony_ci { 4478c2ecf20Sopenharmony_ci unsigned int i; 4488c2ecf20Sopenharmony_ci 4498c2ecf20Sopenharmony_ci for (i = 0; i < sizeof(MGA1064_DAC_regs); i++) { 4508c2ecf20Sopenharmony_ci if ((i != POS1064_XPIXCLKCTRL) && (i != POS1064_XMISCCTRL)) 4518c2ecf20Sopenharmony_ci outDAC1064(minfo, MGA1064_DAC_regs[i], hw->DACreg[i]); 4528c2ecf20Sopenharmony_ci } 4538c2ecf20Sopenharmony_ci } 4548c2ecf20Sopenharmony_ci 4558c2ecf20Sopenharmony_ci DAC1064_global_restore(minfo); 4568c2ecf20Sopenharmony_ci 4578c2ecf20Sopenharmony_ci CRITEND 4588c2ecf20Sopenharmony_ci}; 4598c2ecf20Sopenharmony_ci 4608c2ecf20Sopenharmony_cistatic void DAC1064_restore_2(struct matrox_fb_info *minfo) 4618c2ecf20Sopenharmony_ci{ 4628c2ecf20Sopenharmony_ci#ifdef DEBUG 4638c2ecf20Sopenharmony_ci unsigned int i; 4648c2ecf20Sopenharmony_ci#endif 4658c2ecf20Sopenharmony_ci 4668c2ecf20Sopenharmony_ci DBG(__func__) 4678c2ecf20Sopenharmony_ci 4688c2ecf20Sopenharmony_ci#ifdef DEBUG 4698c2ecf20Sopenharmony_ci dprintk(KERN_DEBUG "DAC1064regs "); 4708c2ecf20Sopenharmony_ci for (i = 0; i < sizeof(MGA1064_DAC_regs); i++) { 4718c2ecf20Sopenharmony_ci dprintk("R%02X=%02X ", MGA1064_DAC_regs[i], minfo->hw.DACreg[i]); 4728c2ecf20Sopenharmony_ci if ((i & 0x7) == 0x7) dprintk(KERN_DEBUG "continuing... "); 4738c2ecf20Sopenharmony_ci } 4748c2ecf20Sopenharmony_ci dprintk(KERN_DEBUG "DAC1064clk "); 4758c2ecf20Sopenharmony_ci for (i = 0; i < 6; i++) 4768c2ecf20Sopenharmony_ci dprintk("C%02X=%02X ", i, minfo->hw.DACclk[i]); 4778c2ecf20Sopenharmony_ci dprintk("\n"); 4788c2ecf20Sopenharmony_ci#endif 4798c2ecf20Sopenharmony_ci} 4808c2ecf20Sopenharmony_ci 4818c2ecf20Sopenharmony_cistatic int m1064_compute(void* out, struct my_timming* m) { 4828c2ecf20Sopenharmony_ci#define minfo ((struct matrox_fb_info*)out) 4838c2ecf20Sopenharmony_ci { 4848c2ecf20Sopenharmony_ci int i; 4858c2ecf20Sopenharmony_ci int tmout; 4868c2ecf20Sopenharmony_ci CRITFLAGS 4878c2ecf20Sopenharmony_ci 4888c2ecf20Sopenharmony_ci DAC1064_setpclk(minfo, m->pixclock); 4898c2ecf20Sopenharmony_ci 4908c2ecf20Sopenharmony_ci CRITBEGIN 4918c2ecf20Sopenharmony_ci 4928c2ecf20Sopenharmony_ci for (i = 0; i < 3; i++) 4938c2ecf20Sopenharmony_ci outDAC1064(minfo, M1064_XPIXPLLCM + i, minfo->hw.DACclk[i]); 4948c2ecf20Sopenharmony_ci for (tmout = 500000; tmout; tmout--) { 4958c2ecf20Sopenharmony_ci if (inDAC1064(minfo, M1064_XPIXPLLSTAT) & 0x40) 4968c2ecf20Sopenharmony_ci break; 4978c2ecf20Sopenharmony_ci udelay(10); 4988c2ecf20Sopenharmony_ci } 4998c2ecf20Sopenharmony_ci 5008c2ecf20Sopenharmony_ci CRITEND 5018c2ecf20Sopenharmony_ci 5028c2ecf20Sopenharmony_ci if (!tmout) 5038c2ecf20Sopenharmony_ci printk(KERN_ERR "matroxfb: Pixel PLL not locked after 5 secs\n"); 5048c2ecf20Sopenharmony_ci } 5058c2ecf20Sopenharmony_ci#undef minfo 5068c2ecf20Sopenharmony_ci return 0; 5078c2ecf20Sopenharmony_ci} 5088c2ecf20Sopenharmony_ci 5098c2ecf20Sopenharmony_cistatic struct matrox_altout m1064 = { 5108c2ecf20Sopenharmony_ci .name = "Primary output", 5118c2ecf20Sopenharmony_ci .compute = m1064_compute, 5128c2ecf20Sopenharmony_ci}; 5138c2ecf20Sopenharmony_ci 5148c2ecf20Sopenharmony_ci#ifdef CONFIG_FB_MATROX_G 5158c2ecf20Sopenharmony_cistatic int g450_compute(void* out, struct my_timming* m) { 5168c2ecf20Sopenharmony_ci#define minfo ((struct matrox_fb_info*)out) 5178c2ecf20Sopenharmony_ci if (m->mnp < 0) { 5188c2ecf20Sopenharmony_ci m->mnp = matroxfb_g450_setclk(minfo, m->pixclock, (m->crtc == MATROXFB_SRC_CRTC1) ? M_PIXEL_PLL_C : M_VIDEO_PLL); 5198c2ecf20Sopenharmony_ci if (m->mnp >= 0) { 5208c2ecf20Sopenharmony_ci m->pixclock = g450_mnp2f(minfo, m->mnp); 5218c2ecf20Sopenharmony_ci } 5228c2ecf20Sopenharmony_ci } 5238c2ecf20Sopenharmony_ci#undef minfo 5248c2ecf20Sopenharmony_ci return 0; 5258c2ecf20Sopenharmony_ci} 5268c2ecf20Sopenharmony_ci 5278c2ecf20Sopenharmony_cistatic struct matrox_altout g450out = { 5288c2ecf20Sopenharmony_ci .name = "Primary output", 5298c2ecf20Sopenharmony_ci .compute = g450_compute, 5308c2ecf20Sopenharmony_ci}; 5318c2ecf20Sopenharmony_ci#endif 5328c2ecf20Sopenharmony_ci 5338c2ecf20Sopenharmony_ci#endif /* NEED_DAC1064 */ 5348c2ecf20Sopenharmony_ci 5358c2ecf20Sopenharmony_ci#ifdef CONFIG_FB_MATROX_MYSTIQUE 5368c2ecf20Sopenharmony_cistatic int MGA1064_init(struct matrox_fb_info *minfo, struct my_timming *m) 5378c2ecf20Sopenharmony_ci{ 5388c2ecf20Sopenharmony_ci struct matrox_hw_state *hw = &minfo->hw; 5398c2ecf20Sopenharmony_ci 5408c2ecf20Sopenharmony_ci DBG(__func__) 5418c2ecf20Sopenharmony_ci 5428c2ecf20Sopenharmony_ci if (DAC1064_init_1(minfo, m)) return 1; 5438c2ecf20Sopenharmony_ci if (matroxfb_vgaHWinit(minfo, m)) return 1; 5448c2ecf20Sopenharmony_ci 5458c2ecf20Sopenharmony_ci hw->MiscOutReg = 0xCB; 5468c2ecf20Sopenharmony_ci if (m->sync & FB_SYNC_HOR_HIGH_ACT) 5478c2ecf20Sopenharmony_ci hw->MiscOutReg &= ~0x40; 5488c2ecf20Sopenharmony_ci if (m->sync & FB_SYNC_VERT_HIGH_ACT) 5498c2ecf20Sopenharmony_ci hw->MiscOutReg &= ~0x80; 5508c2ecf20Sopenharmony_ci if (m->sync & FB_SYNC_COMP_HIGH_ACT) /* should be only FB_SYNC_COMP */ 5518c2ecf20Sopenharmony_ci hw->CRTCEXT[3] |= 0x40; 5528c2ecf20Sopenharmony_ci 5538c2ecf20Sopenharmony_ci if (DAC1064_init_2(minfo, m)) return 1; 5548c2ecf20Sopenharmony_ci return 0; 5558c2ecf20Sopenharmony_ci} 5568c2ecf20Sopenharmony_ci#endif 5578c2ecf20Sopenharmony_ci 5588c2ecf20Sopenharmony_ci#ifdef CONFIG_FB_MATROX_G 5598c2ecf20Sopenharmony_cistatic int MGAG100_init(struct matrox_fb_info *minfo, struct my_timming *m) 5608c2ecf20Sopenharmony_ci{ 5618c2ecf20Sopenharmony_ci struct matrox_hw_state *hw = &minfo->hw; 5628c2ecf20Sopenharmony_ci 5638c2ecf20Sopenharmony_ci DBG(__func__) 5648c2ecf20Sopenharmony_ci 5658c2ecf20Sopenharmony_ci if (DAC1064_init_1(minfo, m)) return 1; 5668c2ecf20Sopenharmony_ci hw->MXoptionReg &= ~0x2000; 5678c2ecf20Sopenharmony_ci if (matroxfb_vgaHWinit(minfo, m)) return 1; 5688c2ecf20Sopenharmony_ci 5698c2ecf20Sopenharmony_ci hw->MiscOutReg = 0xEF; 5708c2ecf20Sopenharmony_ci if (m->sync & FB_SYNC_HOR_HIGH_ACT) 5718c2ecf20Sopenharmony_ci hw->MiscOutReg &= ~0x40; 5728c2ecf20Sopenharmony_ci if (m->sync & FB_SYNC_VERT_HIGH_ACT) 5738c2ecf20Sopenharmony_ci hw->MiscOutReg &= ~0x80; 5748c2ecf20Sopenharmony_ci if (m->sync & FB_SYNC_COMP_HIGH_ACT) /* should be only FB_SYNC_COMP */ 5758c2ecf20Sopenharmony_ci hw->CRTCEXT[3] |= 0x40; 5768c2ecf20Sopenharmony_ci 5778c2ecf20Sopenharmony_ci if (DAC1064_init_2(minfo, m)) return 1; 5788c2ecf20Sopenharmony_ci return 0; 5798c2ecf20Sopenharmony_ci} 5808c2ecf20Sopenharmony_ci#endif /* G */ 5818c2ecf20Sopenharmony_ci 5828c2ecf20Sopenharmony_ci#ifdef CONFIG_FB_MATROX_MYSTIQUE 5838c2ecf20Sopenharmony_cistatic void MGA1064_ramdac_init(struct matrox_fb_info *minfo) 5848c2ecf20Sopenharmony_ci{ 5858c2ecf20Sopenharmony_ci 5868c2ecf20Sopenharmony_ci DBG(__func__) 5878c2ecf20Sopenharmony_ci 5888c2ecf20Sopenharmony_ci /* minfo->features.DAC1064.vco_freq_min = 120000; */ 5898c2ecf20Sopenharmony_ci minfo->features.pll.vco_freq_min = 62000; 5908c2ecf20Sopenharmony_ci minfo->features.pll.ref_freq = 14318; 5918c2ecf20Sopenharmony_ci minfo->features.pll.feed_div_min = 100; 5928c2ecf20Sopenharmony_ci minfo->features.pll.feed_div_max = 127; 5938c2ecf20Sopenharmony_ci minfo->features.pll.in_div_min = 1; 5948c2ecf20Sopenharmony_ci minfo->features.pll.in_div_max = 31; 5958c2ecf20Sopenharmony_ci minfo->features.pll.post_shift_max = 3; 5968c2ecf20Sopenharmony_ci minfo->features.DAC1064.xvrefctrl = DAC1064_XVREFCTRL_EXTERNAL; 5978c2ecf20Sopenharmony_ci /* maybe cmdline MCLK= ?, doc says gclk=44MHz, mclk=66MHz... it was 55/83 with old values */ 5988c2ecf20Sopenharmony_ci DAC1064_setmclk(minfo, DAC1064_OPT_MDIV2 | DAC1064_OPT_GDIV3 | DAC1064_OPT_SCLK_PLL, 133333); 5998c2ecf20Sopenharmony_ci} 6008c2ecf20Sopenharmony_ci#endif 6018c2ecf20Sopenharmony_ci 6028c2ecf20Sopenharmony_ci#ifdef CONFIG_FB_MATROX_G 6038c2ecf20Sopenharmony_ci/* BIOS environ */ 6048c2ecf20Sopenharmony_cistatic int x7AF4 = 0x10; /* flags, maybe 0x10 = SDRAM, 0x00 = SGRAM??? */ 6058c2ecf20Sopenharmony_ci /* G100 wants 0x10, G200 SGRAM does not care... */ 6068c2ecf20Sopenharmony_ci#if 0 6078c2ecf20Sopenharmony_cistatic int def50 = 0; /* reg50, & 0x0F, & 0x3000 (only 0x0000, 0x1000, 0x2000 (0x3000 disallowed and treated as 0) */ 6088c2ecf20Sopenharmony_ci#endif 6098c2ecf20Sopenharmony_ci 6108c2ecf20Sopenharmony_cistatic void MGAG100_progPixClock(const struct matrox_fb_info *minfo, int flags, 6118c2ecf20Sopenharmony_ci int m, int n, int p) 6128c2ecf20Sopenharmony_ci{ 6138c2ecf20Sopenharmony_ci int reg; 6148c2ecf20Sopenharmony_ci int selClk; 6158c2ecf20Sopenharmony_ci int clk; 6168c2ecf20Sopenharmony_ci 6178c2ecf20Sopenharmony_ci DBG(__func__) 6188c2ecf20Sopenharmony_ci 6198c2ecf20Sopenharmony_ci outDAC1064(minfo, M1064_XPIXCLKCTRL, inDAC1064(minfo, M1064_XPIXCLKCTRL) | M1064_XPIXCLKCTRL_DIS | 6208c2ecf20Sopenharmony_ci M1064_XPIXCLKCTRL_PLL_UP); 6218c2ecf20Sopenharmony_ci switch (flags & 3) { 6228c2ecf20Sopenharmony_ci case 0: reg = M1064_XPIXPLLAM; break; 6238c2ecf20Sopenharmony_ci case 1: reg = M1064_XPIXPLLBM; break; 6248c2ecf20Sopenharmony_ci default: reg = M1064_XPIXPLLCM; break; 6258c2ecf20Sopenharmony_ci } 6268c2ecf20Sopenharmony_ci outDAC1064(minfo, reg++, m); 6278c2ecf20Sopenharmony_ci outDAC1064(minfo, reg++, n); 6288c2ecf20Sopenharmony_ci outDAC1064(minfo, reg, p); 6298c2ecf20Sopenharmony_ci selClk = mga_inb(M_MISC_REG_READ) & ~0xC; 6308c2ecf20Sopenharmony_ci /* there should be flags & 0x03 & case 0/1/else */ 6318c2ecf20Sopenharmony_ci /* and we should first select source and after that we should wait for PLL */ 6328c2ecf20Sopenharmony_ci /* and we are waiting for PLL with oscilator disabled... Is it right? */ 6338c2ecf20Sopenharmony_ci switch (flags & 0x03) { 6348c2ecf20Sopenharmony_ci case 0x00: break; 6358c2ecf20Sopenharmony_ci case 0x01: selClk |= 4; break; 6368c2ecf20Sopenharmony_ci default: selClk |= 0x0C; break; 6378c2ecf20Sopenharmony_ci } 6388c2ecf20Sopenharmony_ci mga_outb(M_MISC_REG, selClk); 6398c2ecf20Sopenharmony_ci for (clk = 500000; clk; clk--) { 6408c2ecf20Sopenharmony_ci if (inDAC1064(minfo, M1064_XPIXPLLSTAT) & 0x40) 6418c2ecf20Sopenharmony_ci break; 6428c2ecf20Sopenharmony_ci udelay(10); 6438c2ecf20Sopenharmony_ci } 6448c2ecf20Sopenharmony_ci if (!clk) 6458c2ecf20Sopenharmony_ci printk(KERN_ERR "matroxfb: Pixel PLL%c not locked after usual time\n", (reg-M1064_XPIXPLLAM-2)/4 + 'A'); 6468c2ecf20Sopenharmony_ci selClk = inDAC1064(minfo, M1064_XPIXCLKCTRL) & ~M1064_XPIXCLKCTRL_SRC_MASK; 6478c2ecf20Sopenharmony_ci switch (flags & 0x0C) { 6488c2ecf20Sopenharmony_ci case 0x00: selClk |= M1064_XPIXCLKCTRL_SRC_PCI; break; 6498c2ecf20Sopenharmony_ci case 0x04: selClk |= M1064_XPIXCLKCTRL_SRC_PLL; break; 6508c2ecf20Sopenharmony_ci default: selClk |= M1064_XPIXCLKCTRL_SRC_EXT; break; 6518c2ecf20Sopenharmony_ci } 6528c2ecf20Sopenharmony_ci outDAC1064(minfo, M1064_XPIXCLKCTRL, selClk); 6538c2ecf20Sopenharmony_ci outDAC1064(minfo, M1064_XPIXCLKCTRL, inDAC1064(minfo, M1064_XPIXCLKCTRL) & ~M1064_XPIXCLKCTRL_DIS); 6548c2ecf20Sopenharmony_ci} 6558c2ecf20Sopenharmony_ci 6568c2ecf20Sopenharmony_cistatic void MGAG100_setPixClock(const struct matrox_fb_info *minfo, int flags, 6578c2ecf20Sopenharmony_ci int freq) 6588c2ecf20Sopenharmony_ci{ 6598c2ecf20Sopenharmony_ci unsigned int m, n, p; 6608c2ecf20Sopenharmony_ci 6618c2ecf20Sopenharmony_ci DBG(__func__) 6628c2ecf20Sopenharmony_ci 6638c2ecf20Sopenharmony_ci DAC1064_calcclock(minfo, freq, minfo->max_pixel_clock, &m, &n, &p); 6648c2ecf20Sopenharmony_ci MGAG100_progPixClock(minfo, flags, m, n, p); 6658c2ecf20Sopenharmony_ci} 6668c2ecf20Sopenharmony_ci#endif 6678c2ecf20Sopenharmony_ci 6688c2ecf20Sopenharmony_ci#ifdef CONFIG_FB_MATROX_MYSTIQUE 6698c2ecf20Sopenharmony_cistatic int MGA1064_preinit(struct matrox_fb_info *minfo) 6708c2ecf20Sopenharmony_ci{ 6718c2ecf20Sopenharmony_ci static const int vxres_mystique[] = { 512, 640, 768, 800, 832, 960, 6728c2ecf20Sopenharmony_ci 1024, 1152, 1280, 1600, 1664, 1920, 6738c2ecf20Sopenharmony_ci 2048, 0}; 6748c2ecf20Sopenharmony_ci struct matrox_hw_state *hw = &minfo->hw; 6758c2ecf20Sopenharmony_ci 6768c2ecf20Sopenharmony_ci DBG(__func__) 6778c2ecf20Sopenharmony_ci 6788c2ecf20Sopenharmony_ci /* minfo->capable.cfb4 = 0; ... preinitialized by 0 */ 6798c2ecf20Sopenharmony_ci minfo->capable.text = 1; 6808c2ecf20Sopenharmony_ci minfo->capable.vxres = vxres_mystique; 6818c2ecf20Sopenharmony_ci 6828c2ecf20Sopenharmony_ci minfo->outputs[0].output = &m1064; 6838c2ecf20Sopenharmony_ci minfo->outputs[0].src = minfo->outputs[0].default_src; 6848c2ecf20Sopenharmony_ci minfo->outputs[0].data = minfo; 6858c2ecf20Sopenharmony_ci minfo->outputs[0].mode = MATROXFB_OUTPUT_MODE_MONITOR; 6868c2ecf20Sopenharmony_ci 6878c2ecf20Sopenharmony_ci if (minfo->devflags.noinit) 6888c2ecf20Sopenharmony_ci return 0; /* do not modify settings */ 6898c2ecf20Sopenharmony_ci hw->MXoptionReg &= 0xC0000100; 6908c2ecf20Sopenharmony_ci hw->MXoptionReg |= 0x00094E20; 6918c2ecf20Sopenharmony_ci if (minfo->devflags.novga) 6928c2ecf20Sopenharmony_ci hw->MXoptionReg &= ~0x00000100; 6938c2ecf20Sopenharmony_ci if (minfo->devflags.nobios) 6948c2ecf20Sopenharmony_ci hw->MXoptionReg &= ~0x40000000; 6958c2ecf20Sopenharmony_ci if (minfo->devflags.nopciretry) 6968c2ecf20Sopenharmony_ci hw->MXoptionReg |= 0x20000000; 6978c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION_REG, hw->MXoptionReg); 6988c2ecf20Sopenharmony_ci mga_setr(M_SEQ_INDEX, 0x01, 0x20); 6998c2ecf20Sopenharmony_ci mga_outl(M_CTLWTST, 0x00000000); 7008c2ecf20Sopenharmony_ci udelay(200); 7018c2ecf20Sopenharmony_ci mga_outl(M_MACCESS, 0x00008000); 7028c2ecf20Sopenharmony_ci udelay(100); 7038c2ecf20Sopenharmony_ci mga_outl(M_MACCESS, 0x0000C000); 7048c2ecf20Sopenharmony_ci return 0; 7058c2ecf20Sopenharmony_ci} 7068c2ecf20Sopenharmony_ci 7078c2ecf20Sopenharmony_cistatic void MGA1064_reset(struct matrox_fb_info *minfo) 7088c2ecf20Sopenharmony_ci{ 7098c2ecf20Sopenharmony_ci 7108c2ecf20Sopenharmony_ci DBG(__func__); 7118c2ecf20Sopenharmony_ci 7128c2ecf20Sopenharmony_ci MGA1064_ramdac_init(minfo); 7138c2ecf20Sopenharmony_ci} 7148c2ecf20Sopenharmony_ci#endif 7158c2ecf20Sopenharmony_ci 7168c2ecf20Sopenharmony_ci#ifdef CONFIG_FB_MATROX_G 7178c2ecf20Sopenharmony_cistatic void g450_mclk_init(struct matrox_fb_info *minfo) 7188c2ecf20Sopenharmony_ci{ 7198c2ecf20Sopenharmony_ci /* switch all clocks to PCI source */ 7208c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION_REG, minfo->hw.MXoptionReg | 4); 7218c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION3_REG, minfo->values.reg.opt3 & ~0x00300C03); 7228c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION_REG, minfo->hw.MXoptionReg); 7238c2ecf20Sopenharmony_ci 7248c2ecf20Sopenharmony_ci if (((minfo->values.reg.opt3 & 0x000003) == 0x000003) || 7258c2ecf20Sopenharmony_ci ((minfo->values.reg.opt3 & 0x000C00) == 0x000C00) || 7268c2ecf20Sopenharmony_ci ((minfo->values.reg.opt3 & 0x300000) == 0x300000)) { 7278c2ecf20Sopenharmony_ci matroxfb_g450_setclk(minfo, minfo->values.pll.video, M_VIDEO_PLL); 7288c2ecf20Sopenharmony_ci } else { 7298c2ecf20Sopenharmony_ci unsigned long flags; 7308c2ecf20Sopenharmony_ci unsigned int pwr; 7318c2ecf20Sopenharmony_ci 7328c2ecf20Sopenharmony_ci matroxfb_DAC_lock_irqsave(flags); 7338c2ecf20Sopenharmony_ci pwr = inDAC1064(minfo, M1064_XPWRCTRL) & ~0x02; 7348c2ecf20Sopenharmony_ci outDAC1064(minfo, M1064_XPWRCTRL, pwr); 7358c2ecf20Sopenharmony_ci matroxfb_DAC_unlock_irqrestore(flags); 7368c2ecf20Sopenharmony_ci } 7378c2ecf20Sopenharmony_ci matroxfb_g450_setclk(minfo, minfo->values.pll.system, M_SYSTEM_PLL); 7388c2ecf20Sopenharmony_ci 7398c2ecf20Sopenharmony_ci /* switch clocks to their real PLL source(s) */ 7408c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION_REG, minfo->hw.MXoptionReg | 4); 7418c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION3_REG, minfo->values.reg.opt3); 7428c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION_REG, minfo->hw.MXoptionReg); 7438c2ecf20Sopenharmony_ci 7448c2ecf20Sopenharmony_ci} 7458c2ecf20Sopenharmony_ci 7468c2ecf20Sopenharmony_cistatic void g450_memory_init(struct matrox_fb_info *minfo) 7478c2ecf20Sopenharmony_ci{ 7488c2ecf20Sopenharmony_ci /* disable memory refresh */ 7498c2ecf20Sopenharmony_ci minfo->hw.MXoptionReg &= ~0x001F8000; 7508c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION_REG, minfo->hw.MXoptionReg); 7518c2ecf20Sopenharmony_ci 7528c2ecf20Sopenharmony_ci /* set memory interface parameters */ 7538c2ecf20Sopenharmony_ci minfo->hw.MXoptionReg &= ~0x00207E00; 7548c2ecf20Sopenharmony_ci minfo->hw.MXoptionReg |= 0x00207E00 & minfo->values.reg.opt; 7558c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION_REG, minfo->hw.MXoptionReg); 7568c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION2_REG, minfo->values.reg.opt2); 7578c2ecf20Sopenharmony_ci 7588c2ecf20Sopenharmony_ci mga_outl(M_CTLWTST, minfo->values.reg.mctlwtst); 7598c2ecf20Sopenharmony_ci 7608c2ecf20Sopenharmony_ci /* first set up memory interface with disabled memory interface clocks */ 7618c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_MEMMISC_REG, minfo->values.reg.memmisc & ~0x80000000U); 7628c2ecf20Sopenharmony_ci mga_outl(M_MEMRDBK, minfo->values.reg.memrdbk); 7638c2ecf20Sopenharmony_ci mga_outl(M_MACCESS, minfo->values.reg.maccess); 7648c2ecf20Sopenharmony_ci /* start memory clocks */ 7658c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_MEMMISC_REG, minfo->values.reg.memmisc | 0x80000000U); 7668c2ecf20Sopenharmony_ci 7678c2ecf20Sopenharmony_ci udelay(200); 7688c2ecf20Sopenharmony_ci 7698c2ecf20Sopenharmony_ci if (minfo->values.memory.ddr && (!minfo->values.memory.emrswen || !minfo->values.memory.dll)) { 7708c2ecf20Sopenharmony_ci mga_outl(M_MEMRDBK, minfo->values.reg.memrdbk & ~0x1000); 7718c2ecf20Sopenharmony_ci } 7728c2ecf20Sopenharmony_ci mga_outl(M_MACCESS, minfo->values.reg.maccess | 0x8000); 7738c2ecf20Sopenharmony_ci 7748c2ecf20Sopenharmony_ci udelay(200); 7758c2ecf20Sopenharmony_ci 7768c2ecf20Sopenharmony_ci minfo->hw.MXoptionReg |= 0x001F8000 & minfo->values.reg.opt; 7778c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION_REG, minfo->hw.MXoptionReg); 7788c2ecf20Sopenharmony_ci 7798c2ecf20Sopenharmony_ci /* value is written to memory chips only if old != new */ 7808c2ecf20Sopenharmony_ci mga_outl(M_PLNWT, 0); 7818c2ecf20Sopenharmony_ci mga_outl(M_PLNWT, ~0); 7828c2ecf20Sopenharmony_ci 7838c2ecf20Sopenharmony_ci if (minfo->values.reg.mctlwtst != minfo->values.reg.mctlwtst_core) { 7848c2ecf20Sopenharmony_ci mga_outl(M_CTLWTST, minfo->values.reg.mctlwtst_core); 7858c2ecf20Sopenharmony_ci } 7868c2ecf20Sopenharmony_ci 7878c2ecf20Sopenharmony_ci} 7888c2ecf20Sopenharmony_ci 7898c2ecf20Sopenharmony_cistatic void g450_preinit(struct matrox_fb_info *minfo) 7908c2ecf20Sopenharmony_ci{ 7918c2ecf20Sopenharmony_ci u_int32_t c2ctl; 7928c2ecf20Sopenharmony_ci u_int8_t curctl; 7938c2ecf20Sopenharmony_ci u_int8_t c1ctl; 7948c2ecf20Sopenharmony_ci 7958c2ecf20Sopenharmony_ci /* minfo->hw.MXoptionReg = minfo->values.reg.opt; */ 7968c2ecf20Sopenharmony_ci minfo->hw.MXoptionReg &= 0xC0000100; 7978c2ecf20Sopenharmony_ci minfo->hw.MXoptionReg |= 0x00000020; 7988c2ecf20Sopenharmony_ci if (minfo->devflags.novga) 7998c2ecf20Sopenharmony_ci minfo->hw.MXoptionReg &= ~0x00000100; 8008c2ecf20Sopenharmony_ci if (minfo->devflags.nobios) 8018c2ecf20Sopenharmony_ci minfo->hw.MXoptionReg &= ~0x40000000; 8028c2ecf20Sopenharmony_ci if (minfo->devflags.nopciretry) 8038c2ecf20Sopenharmony_ci minfo->hw.MXoptionReg |= 0x20000000; 8048c2ecf20Sopenharmony_ci minfo->hw.MXoptionReg |= minfo->values.reg.opt & 0x03400040; 8058c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION_REG, minfo->hw.MXoptionReg); 8068c2ecf20Sopenharmony_ci 8078c2ecf20Sopenharmony_ci /* Init system clocks */ 8088c2ecf20Sopenharmony_ci 8098c2ecf20Sopenharmony_ci /* stop crtc2 */ 8108c2ecf20Sopenharmony_ci c2ctl = mga_inl(M_C2CTL); 8118c2ecf20Sopenharmony_ci mga_outl(M_C2CTL, c2ctl & ~1); 8128c2ecf20Sopenharmony_ci /* stop cursor */ 8138c2ecf20Sopenharmony_ci curctl = inDAC1064(minfo, M1064_XCURCTRL); 8148c2ecf20Sopenharmony_ci outDAC1064(minfo, M1064_XCURCTRL, 0); 8158c2ecf20Sopenharmony_ci /* stop crtc1 */ 8168c2ecf20Sopenharmony_ci c1ctl = mga_readr(M_SEQ_INDEX, 1); 8178c2ecf20Sopenharmony_ci mga_setr(M_SEQ_INDEX, 1, c1ctl | 0x20); 8188c2ecf20Sopenharmony_ci 8198c2ecf20Sopenharmony_ci g450_mclk_init(minfo); 8208c2ecf20Sopenharmony_ci g450_memory_init(minfo); 8218c2ecf20Sopenharmony_ci 8228c2ecf20Sopenharmony_ci /* set legacy VGA clock sources for DOSEmu or VMware... */ 8238c2ecf20Sopenharmony_ci matroxfb_g450_setclk(minfo, 25175, M_PIXEL_PLL_A); 8248c2ecf20Sopenharmony_ci matroxfb_g450_setclk(minfo, 28322, M_PIXEL_PLL_B); 8258c2ecf20Sopenharmony_ci 8268c2ecf20Sopenharmony_ci /* restore crtc1 */ 8278c2ecf20Sopenharmony_ci mga_setr(M_SEQ_INDEX, 1, c1ctl); 8288c2ecf20Sopenharmony_ci 8298c2ecf20Sopenharmony_ci /* restore cursor */ 8308c2ecf20Sopenharmony_ci outDAC1064(minfo, M1064_XCURCTRL, curctl); 8318c2ecf20Sopenharmony_ci 8328c2ecf20Sopenharmony_ci /* restore crtc2 */ 8338c2ecf20Sopenharmony_ci mga_outl(M_C2CTL, c2ctl); 8348c2ecf20Sopenharmony_ci 8358c2ecf20Sopenharmony_ci return; 8368c2ecf20Sopenharmony_ci} 8378c2ecf20Sopenharmony_ci 8388c2ecf20Sopenharmony_cistatic int MGAG100_preinit(struct matrox_fb_info *minfo) 8398c2ecf20Sopenharmony_ci{ 8408c2ecf20Sopenharmony_ci static const int vxres_g100[] = { 512, 640, 768, 800, 832, 960, 8418c2ecf20Sopenharmony_ci 1024, 1152, 1280, 1600, 1664, 1920, 8428c2ecf20Sopenharmony_ci 2048, 0}; 8438c2ecf20Sopenharmony_ci struct matrox_hw_state *hw = &minfo->hw; 8448c2ecf20Sopenharmony_ci 8458c2ecf20Sopenharmony_ci u_int32_t reg50; 8468c2ecf20Sopenharmony_ci#if 0 8478c2ecf20Sopenharmony_ci u_int32_t q; 8488c2ecf20Sopenharmony_ci#endif 8498c2ecf20Sopenharmony_ci 8508c2ecf20Sopenharmony_ci DBG(__func__) 8518c2ecf20Sopenharmony_ci 8528c2ecf20Sopenharmony_ci /* there are some instabilities if in_div > 19 && vco < 61000 */ 8538c2ecf20Sopenharmony_ci if (minfo->devflags.g450dac) { 8548c2ecf20Sopenharmony_ci minfo->features.pll.vco_freq_min = 130000; /* my sample: >118 */ 8558c2ecf20Sopenharmony_ci } else { 8568c2ecf20Sopenharmony_ci minfo->features.pll.vco_freq_min = 62000; 8578c2ecf20Sopenharmony_ci } 8588c2ecf20Sopenharmony_ci if (!minfo->features.pll.ref_freq) { 8598c2ecf20Sopenharmony_ci minfo->features.pll.ref_freq = 27000; 8608c2ecf20Sopenharmony_ci } 8618c2ecf20Sopenharmony_ci minfo->features.pll.feed_div_min = 7; 8628c2ecf20Sopenharmony_ci minfo->features.pll.feed_div_max = 127; 8638c2ecf20Sopenharmony_ci minfo->features.pll.in_div_min = 1; 8648c2ecf20Sopenharmony_ci minfo->features.pll.in_div_max = 31; 8658c2ecf20Sopenharmony_ci minfo->features.pll.post_shift_max = 3; 8668c2ecf20Sopenharmony_ci minfo->features.DAC1064.xvrefctrl = DAC1064_XVREFCTRL_G100_DEFAULT; 8678c2ecf20Sopenharmony_ci /* minfo->capable.cfb4 = 0; ... preinitialized by 0 */ 8688c2ecf20Sopenharmony_ci minfo->capable.text = 1; 8698c2ecf20Sopenharmony_ci minfo->capable.vxres = vxres_g100; 8708c2ecf20Sopenharmony_ci minfo->capable.plnwt = minfo->devflags.accelerator == FB_ACCEL_MATROX_MGAG100 8718c2ecf20Sopenharmony_ci ? minfo->devflags.sgram : 1; 8728c2ecf20Sopenharmony_ci 8738c2ecf20Sopenharmony_ci if (minfo->devflags.g450dac) { 8748c2ecf20Sopenharmony_ci minfo->outputs[0].output = &g450out; 8758c2ecf20Sopenharmony_ci } else { 8768c2ecf20Sopenharmony_ci minfo->outputs[0].output = &m1064; 8778c2ecf20Sopenharmony_ci } 8788c2ecf20Sopenharmony_ci minfo->outputs[0].src = minfo->outputs[0].default_src; 8798c2ecf20Sopenharmony_ci minfo->outputs[0].data = minfo; 8808c2ecf20Sopenharmony_ci minfo->outputs[0].mode = MATROXFB_OUTPUT_MODE_MONITOR; 8818c2ecf20Sopenharmony_ci 8828c2ecf20Sopenharmony_ci if (minfo->devflags.g450dac) { 8838c2ecf20Sopenharmony_ci /* we must do this always, BIOS does not do it for us 8848c2ecf20Sopenharmony_ci and accelerator dies without it */ 8858c2ecf20Sopenharmony_ci mga_outl(0x1C0C, 0); 8868c2ecf20Sopenharmony_ci } 8878c2ecf20Sopenharmony_ci if (minfo->devflags.noinit) 8888c2ecf20Sopenharmony_ci return 0; 8898c2ecf20Sopenharmony_ci if (minfo->devflags.g450dac) { 8908c2ecf20Sopenharmony_ci g450_preinit(minfo); 8918c2ecf20Sopenharmony_ci return 0; 8928c2ecf20Sopenharmony_ci } 8938c2ecf20Sopenharmony_ci hw->MXoptionReg &= 0xC0000100; 8948c2ecf20Sopenharmony_ci hw->MXoptionReg |= 0x00000020; 8958c2ecf20Sopenharmony_ci if (minfo->devflags.novga) 8968c2ecf20Sopenharmony_ci hw->MXoptionReg &= ~0x00000100; 8978c2ecf20Sopenharmony_ci if (minfo->devflags.nobios) 8988c2ecf20Sopenharmony_ci hw->MXoptionReg &= ~0x40000000; 8998c2ecf20Sopenharmony_ci if (minfo->devflags.nopciretry) 9008c2ecf20Sopenharmony_ci hw->MXoptionReg |= 0x20000000; 9018c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION_REG, hw->MXoptionReg); 9028c2ecf20Sopenharmony_ci DAC1064_setmclk(minfo, DAC1064_OPT_MDIV2 | DAC1064_OPT_GDIV3 | DAC1064_OPT_SCLK_PCI, 133333); 9038c2ecf20Sopenharmony_ci 9048c2ecf20Sopenharmony_ci if (minfo->devflags.accelerator == FB_ACCEL_MATROX_MGAG100) { 9058c2ecf20Sopenharmony_ci pci_read_config_dword(minfo->pcidev, PCI_OPTION2_REG, ®50); 9068c2ecf20Sopenharmony_ci reg50 &= ~0x3000; 9078c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION2_REG, reg50); 9088c2ecf20Sopenharmony_ci 9098c2ecf20Sopenharmony_ci hw->MXoptionReg |= 0x1080; 9108c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION_REG, hw->MXoptionReg); 9118c2ecf20Sopenharmony_ci mga_outl(M_CTLWTST, minfo->values.reg.mctlwtst); 9128c2ecf20Sopenharmony_ci udelay(100); 9138c2ecf20Sopenharmony_ci mga_outb(0x1C05, 0x00); 9148c2ecf20Sopenharmony_ci mga_outb(0x1C05, 0x80); 9158c2ecf20Sopenharmony_ci udelay(100); 9168c2ecf20Sopenharmony_ci mga_outb(0x1C05, 0x40); 9178c2ecf20Sopenharmony_ci mga_outb(0x1C05, 0xC0); 9188c2ecf20Sopenharmony_ci udelay(100); 9198c2ecf20Sopenharmony_ci reg50 &= ~0xFF; 9208c2ecf20Sopenharmony_ci reg50 |= 0x07; 9218c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION2_REG, reg50); 9228c2ecf20Sopenharmony_ci /* it should help with G100 */ 9238c2ecf20Sopenharmony_ci mga_outb(M_GRAPHICS_INDEX, 6); 9248c2ecf20Sopenharmony_ci mga_outb(M_GRAPHICS_DATA, (mga_inb(M_GRAPHICS_DATA) & 3) | 4); 9258c2ecf20Sopenharmony_ci mga_setr(M_EXTVGA_INDEX, 0x03, 0x81); 9268c2ecf20Sopenharmony_ci mga_setr(M_EXTVGA_INDEX, 0x04, 0x00); 9278c2ecf20Sopenharmony_ci mga_writeb(minfo->video.vbase, 0x0000, 0xAA); 9288c2ecf20Sopenharmony_ci mga_writeb(minfo->video.vbase, 0x0800, 0x55); 9298c2ecf20Sopenharmony_ci mga_writeb(minfo->video.vbase, 0x4000, 0x55); 9308c2ecf20Sopenharmony_ci#if 0 9318c2ecf20Sopenharmony_ci if (mga_readb(minfo->video.vbase, 0x0000) != 0xAA) { 9328c2ecf20Sopenharmony_ci hw->MXoptionReg &= ~0x1000; 9338c2ecf20Sopenharmony_ci } 9348c2ecf20Sopenharmony_ci#endif 9358c2ecf20Sopenharmony_ci hw->MXoptionReg |= 0x00078020; 9368c2ecf20Sopenharmony_ci } else if (minfo->devflags.accelerator == FB_ACCEL_MATROX_MGAG200) { 9378c2ecf20Sopenharmony_ci pci_read_config_dword(minfo->pcidev, PCI_OPTION2_REG, ®50); 9388c2ecf20Sopenharmony_ci reg50 &= ~0x3000; 9398c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION2_REG, reg50); 9408c2ecf20Sopenharmony_ci 9418c2ecf20Sopenharmony_ci if (minfo->devflags.memtype == -1) 9428c2ecf20Sopenharmony_ci hw->MXoptionReg |= minfo->values.reg.opt & 0x1C00; 9438c2ecf20Sopenharmony_ci else 9448c2ecf20Sopenharmony_ci hw->MXoptionReg |= (minfo->devflags.memtype & 7) << 10; 9458c2ecf20Sopenharmony_ci if (minfo->devflags.sgram) 9468c2ecf20Sopenharmony_ci hw->MXoptionReg |= 0x4000; 9478c2ecf20Sopenharmony_ci mga_outl(M_CTLWTST, minfo->values.reg.mctlwtst); 9488c2ecf20Sopenharmony_ci mga_outl(M_MEMRDBK, minfo->values.reg.memrdbk); 9498c2ecf20Sopenharmony_ci udelay(200); 9508c2ecf20Sopenharmony_ci mga_outl(M_MACCESS, 0x00000000); 9518c2ecf20Sopenharmony_ci mga_outl(M_MACCESS, 0x00008000); 9528c2ecf20Sopenharmony_ci udelay(100); 9538c2ecf20Sopenharmony_ci mga_outw(M_MEMRDBK, minfo->values.reg.memrdbk); 9548c2ecf20Sopenharmony_ci hw->MXoptionReg |= 0x00078020; 9558c2ecf20Sopenharmony_ci } else { 9568c2ecf20Sopenharmony_ci pci_read_config_dword(minfo->pcidev, PCI_OPTION2_REG, ®50); 9578c2ecf20Sopenharmony_ci reg50 &= ~0x00000100; 9588c2ecf20Sopenharmony_ci reg50 |= 0x00000000; 9598c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION2_REG, reg50); 9608c2ecf20Sopenharmony_ci 9618c2ecf20Sopenharmony_ci if (minfo->devflags.memtype == -1) 9628c2ecf20Sopenharmony_ci hw->MXoptionReg |= minfo->values.reg.opt & 0x1C00; 9638c2ecf20Sopenharmony_ci else 9648c2ecf20Sopenharmony_ci hw->MXoptionReg |= (minfo->devflags.memtype & 7) << 10; 9658c2ecf20Sopenharmony_ci if (minfo->devflags.sgram) 9668c2ecf20Sopenharmony_ci hw->MXoptionReg |= 0x4000; 9678c2ecf20Sopenharmony_ci mga_outl(M_CTLWTST, minfo->values.reg.mctlwtst); 9688c2ecf20Sopenharmony_ci mga_outl(M_MEMRDBK, minfo->values.reg.memrdbk); 9698c2ecf20Sopenharmony_ci udelay(200); 9708c2ecf20Sopenharmony_ci mga_outl(M_MACCESS, 0x00000000); 9718c2ecf20Sopenharmony_ci mga_outl(M_MACCESS, 0x00008000); 9728c2ecf20Sopenharmony_ci udelay(100); 9738c2ecf20Sopenharmony_ci mga_outl(M_MEMRDBK, minfo->values.reg.memrdbk); 9748c2ecf20Sopenharmony_ci hw->MXoptionReg |= 0x00040020; 9758c2ecf20Sopenharmony_ci } 9768c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION_REG, hw->MXoptionReg); 9778c2ecf20Sopenharmony_ci return 0; 9788c2ecf20Sopenharmony_ci} 9798c2ecf20Sopenharmony_ci 9808c2ecf20Sopenharmony_cistatic void MGAG100_reset(struct matrox_fb_info *minfo) 9818c2ecf20Sopenharmony_ci{ 9828c2ecf20Sopenharmony_ci u_int8_t b; 9838c2ecf20Sopenharmony_ci struct matrox_hw_state *hw = &minfo->hw; 9848c2ecf20Sopenharmony_ci 9858c2ecf20Sopenharmony_ci DBG(__func__) 9868c2ecf20Sopenharmony_ci 9878c2ecf20Sopenharmony_ci { 9888c2ecf20Sopenharmony_ci#ifdef G100_BROKEN_IBM_82351 9898c2ecf20Sopenharmony_ci u_int32_t d; 9908c2ecf20Sopenharmony_ci 9918c2ecf20Sopenharmony_ci find 1014/22 (IBM/82351); /* if found and bridging Matrox, do some strange stuff */ 9928c2ecf20Sopenharmony_ci pci_read_config_byte(ibm, PCI_SECONDARY_BUS, &b); 9938c2ecf20Sopenharmony_ci if (b == minfo->pcidev->bus->number) { 9948c2ecf20Sopenharmony_ci pci_write_config_byte(ibm, PCI_COMMAND+1, 0); /* disable back-to-back & SERR */ 9958c2ecf20Sopenharmony_ci pci_write_config_byte(ibm, 0x41, 0xF4); /* ??? */ 9968c2ecf20Sopenharmony_ci pci_write_config_byte(ibm, PCI_IO_BASE, 0xF0); /* ??? */ 9978c2ecf20Sopenharmony_ci pci_write_config_byte(ibm, PCI_IO_LIMIT, 0x00); /* ??? */ 9988c2ecf20Sopenharmony_ci } 9998c2ecf20Sopenharmony_ci#endif 10008c2ecf20Sopenharmony_ci if (!minfo->devflags.noinit) { 10018c2ecf20Sopenharmony_ci if (x7AF4 & 8) { 10028c2ecf20Sopenharmony_ci hw->MXoptionReg |= 0x40; /* FIXME... */ 10038c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION_REG, hw->MXoptionReg); 10048c2ecf20Sopenharmony_ci } 10058c2ecf20Sopenharmony_ci mga_setr(M_EXTVGA_INDEX, 0x06, 0x00); 10068c2ecf20Sopenharmony_ci } 10078c2ecf20Sopenharmony_ci } 10088c2ecf20Sopenharmony_ci if (minfo->devflags.g450dac) { 10098c2ecf20Sopenharmony_ci /* either leave MCLK as is... or they were set in preinit */ 10108c2ecf20Sopenharmony_ci hw->DACclk[3] = inDAC1064(minfo, DAC1064_XSYSPLLM); 10118c2ecf20Sopenharmony_ci hw->DACclk[4] = inDAC1064(minfo, DAC1064_XSYSPLLN); 10128c2ecf20Sopenharmony_ci hw->DACclk[5] = inDAC1064(minfo, DAC1064_XSYSPLLP); 10138c2ecf20Sopenharmony_ci } else { 10148c2ecf20Sopenharmony_ci DAC1064_setmclk(minfo, DAC1064_OPT_RESERVED | DAC1064_OPT_MDIV2 | DAC1064_OPT_GDIV1 | DAC1064_OPT_SCLK_PLL, 133333); 10158c2ecf20Sopenharmony_ci } 10168c2ecf20Sopenharmony_ci if (minfo->devflags.accelerator == FB_ACCEL_MATROX_MGAG400) { 10178c2ecf20Sopenharmony_ci if (minfo->devflags.dfp_type == -1) { 10188c2ecf20Sopenharmony_ci minfo->devflags.dfp_type = inDAC1064(minfo, 0x1F); 10198c2ecf20Sopenharmony_ci } 10208c2ecf20Sopenharmony_ci } 10218c2ecf20Sopenharmony_ci if (minfo->devflags.noinit) 10228c2ecf20Sopenharmony_ci return; 10238c2ecf20Sopenharmony_ci if (minfo->devflags.g450dac) { 10248c2ecf20Sopenharmony_ci } else { 10258c2ecf20Sopenharmony_ci MGAG100_setPixClock(minfo, 4, 25175); 10268c2ecf20Sopenharmony_ci MGAG100_setPixClock(minfo, 5, 28322); 10278c2ecf20Sopenharmony_ci if (x7AF4 & 0x10) { 10288c2ecf20Sopenharmony_ci b = inDAC1064(minfo, M1064_XGENIODATA) & ~1; 10298c2ecf20Sopenharmony_ci outDAC1064(minfo, M1064_XGENIODATA, b); 10308c2ecf20Sopenharmony_ci b = inDAC1064(minfo, M1064_XGENIOCTRL) | 1; 10318c2ecf20Sopenharmony_ci outDAC1064(minfo, M1064_XGENIOCTRL, b); 10328c2ecf20Sopenharmony_ci } 10338c2ecf20Sopenharmony_ci } 10348c2ecf20Sopenharmony_ci} 10358c2ecf20Sopenharmony_ci#endif 10368c2ecf20Sopenharmony_ci 10378c2ecf20Sopenharmony_ci#ifdef CONFIG_FB_MATROX_MYSTIQUE 10388c2ecf20Sopenharmony_cistatic void MGA1064_restore(struct matrox_fb_info *minfo) 10398c2ecf20Sopenharmony_ci{ 10408c2ecf20Sopenharmony_ci int i; 10418c2ecf20Sopenharmony_ci struct matrox_hw_state *hw = &minfo->hw; 10428c2ecf20Sopenharmony_ci 10438c2ecf20Sopenharmony_ci CRITFLAGS 10448c2ecf20Sopenharmony_ci 10458c2ecf20Sopenharmony_ci DBG(__func__) 10468c2ecf20Sopenharmony_ci 10478c2ecf20Sopenharmony_ci CRITBEGIN 10488c2ecf20Sopenharmony_ci 10498c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION_REG, hw->MXoptionReg); 10508c2ecf20Sopenharmony_ci mga_outb(M_IEN, 0x00); 10518c2ecf20Sopenharmony_ci mga_outb(M_CACHEFLUSH, 0x00); 10528c2ecf20Sopenharmony_ci 10538c2ecf20Sopenharmony_ci CRITEND 10548c2ecf20Sopenharmony_ci 10558c2ecf20Sopenharmony_ci DAC1064_restore_1(minfo); 10568c2ecf20Sopenharmony_ci matroxfb_vgaHWrestore(minfo); 10578c2ecf20Sopenharmony_ci minfo->crtc1.panpos = -1; 10588c2ecf20Sopenharmony_ci for (i = 0; i < 6; i++) 10598c2ecf20Sopenharmony_ci mga_setr(M_EXTVGA_INDEX, i, hw->CRTCEXT[i]); 10608c2ecf20Sopenharmony_ci DAC1064_restore_2(minfo); 10618c2ecf20Sopenharmony_ci} 10628c2ecf20Sopenharmony_ci#endif 10638c2ecf20Sopenharmony_ci 10648c2ecf20Sopenharmony_ci#ifdef CONFIG_FB_MATROX_G 10658c2ecf20Sopenharmony_cistatic void MGAG100_restore(struct matrox_fb_info *minfo) 10668c2ecf20Sopenharmony_ci{ 10678c2ecf20Sopenharmony_ci int i; 10688c2ecf20Sopenharmony_ci struct matrox_hw_state *hw = &minfo->hw; 10698c2ecf20Sopenharmony_ci 10708c2ecf20Sopenharmony_ci CRITFLAGS 10718c2ecf20Sopenharmony_ci 10728c2ecf20Sopenharmony_ci DBG(__func__) 10738c2ecf20Sopenharmony_ci 10748c2ecf20Sopenharmony_ci CRITBEGIN 10758c2ecf20Sopenharmony_ci 10768c2ecf20Sopenharmony_ci pci_write_config_dword(minfo->pcidev, PCI_OPTION_REG, hw->MXoptionReg); 10778c2ecf20Sopenharmony_ci CRITEND 10788c2ecf20Sopenharmony_ci 10798c2ecf20Sopenharmony_ci DAC1064_restore_1(minfo); 10808c2ecf20Sopenharmony_ci matroxfb_vgaHWrestore(minfo); 10818c2ecf20Sopenharmony_ci if (minfo->devflags.support32MB) 10828c2ecf20Sopenharmony_ci mga_setr(M_EXTVGA_INDEX, 8, hw->CRTCEXT[8]); 10838c2ecf20Sopenharmony_ci minfo->crtc1.panpos = -1; 10848c2ecf20Sopenharmony_ci for (i = 0; i < 6; i++) 10858c2ecf20Sopenharmony_ci mga_setr(M_EXTVGA_INDEX, i, hw->CRTCEXT[i]); 10868c2ecf20Sopenharmony_ci DAC1064_restore_2(minfo); 10878c2ecf20Sopenharmony_ci} 10888c2ecf20Sopenharmony_ci#endif 10898c2ecf20Sopenharmony_ci 10908c2ecf20Sopenharmony_ci#ifdef CONFIG_FB_MATROX_MYSTIQUE 10918c2ecf20Sopenharmony_cistruct matrox_switch matrox_mystique = { 10928c2ecf20Sopenharmony_ci .preinit = MGA1064_preinit, 10938c2ecf20Sopenharmony_ci .reset = MGA1064_reset, 10948c2ecf20Sopenharmony_ci .init = MGA1064_init, 10958c2ecf20Sopenharmony_ci .restore = MGA1064_restore, 10968c2ecf20Sopenharmony_ci}; 10978c2ecf20Sopenharmony_ciEXPORT_SYMBOL(matrox_mystique); 10988c2ecf20Sopenharmony_ci#endif 10998c2ecf20Sopenharmony_ci 11008c2ecf20Sopenharmony_ci#ifdef CONFIG_FB_MATROX_G 11018c2ecf20Sopenharmony_cistruct matrox_switch matrox_G100 = { 11028c2ecf20Sopenharmony_ci .preinit = MGAG100_preinit, 11038c2ecf20Sopenharmony_ci .reset = MGAG100_reset, 11048c2ecf20Sopenharmony_ci .init = MGAG100_init, 11058c2ecf20Sopenharmony_ci .restore = MGAG100_restore, 11068c2ecf20Sopenharmony_ci}; 11078c2ecf20Sopenharmony_ciEXPORT_SYMBOL(matrox_G100); 11088c2ecf20Sopenharmony_ci#endif 11098c2ecf20Sopenharmony_ci 11108c2ecf20Sopenharmony_ci#ifdef NEED_DAC1064 11118c2ecf20Sopenharmony_ciEXPORT_SYMBOL(DAC1064_global_init); 11128c2ecf20Sopenharmony_ciEXPORT_SYMBOL(DAC1064_global_restore); 11138c2ecf20Sopenharmony_ci#endif 11148c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL"); 1115