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, &reg50);
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, &reg50);
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, &reg50);
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