blob: 274dacd54bb82d269f982333ad1c2bdf79e1d5fc [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/* $XFree86$ */
2/* $XdotOrg$ */
3/*
4 * Mode initializing code (CRT2 section)
5 * for SiS 300/305/540/630/730 and
6 * SiS 315/550/650/M650/651/661FX/M661xX/740/741(GX)/M741/330/660/M660/760/M760
7 * (Universal module for Linux kernel framebuffer and XFree86/X.org 4.x)
8 *
9 * Copyright (C) 2001-2004 by Thomas Winischhofer, Vienna, Austria
10 *
11 * If distributed as part of the Linux kernel, the following license terms
12 * apply:
13 *
14 * * This program is free software; you can redistribute it and/or modify
15 * * it under the terms of the GNU General Public License as published by
16 * * the Free Software Foundation; either version 2 of the named License,
17 * * or any later version.
18 * *
19 * * This program is distributed in the hope that it will be useful,
20 * * but WITHOUT ANY WARRANTY; without even the implied warranty of
21 * * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
22 * * GNU General Public License for more details.
23 * *
24 * * You should have received a copy of the GNU General Public License
25 * * along with this program; if not, write to the Free Software
26 * * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA
27 *
28 * Otherwise, the following license terms apply:
29 *
30 * * Redistribution and use in source and binary forms, with or without
31 * * modification, are permitted provided that the following conditions
32 * * are met:
33 * * 1) Redistributions of source code must retain the above copyright
34 * * notice, this list of conditions and the following disclaimer.
35 * * 2) Redistributions in binary form must reproduce the above copyright
36 * * notice, this list of conditions and the following disclaimer in the
37 * * documentation and/or other materials provided with the distribution.
38 * * 3) The name of the author may not be used to endorse or promote products
39 * * derived from this software without specific prior written permission.
40 * *
41 * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESSED OR
42 * * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
43 * * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
44 * * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
45 * * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
46 * * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
47 * * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
48 * * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
49 * * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
50 * * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
51 *
52 * Author: Thomas Winischhofer <thomas@winischhofer.net>
53 *
54 * Formerly based on non-functional code-fragements for 300 series by SiS, Inc.
55 * Used by permission.
56 *
57 * TW says: This code looks awful, I know. But please don't do anything about
58 * this otherwise debugging will be hell.
59 * The code is extremely fragile as regards the different chipsets, different
60 * video bridges and combinations thereof. If anything is changed, extreme
61 * care has to be taken that that change doesn't break it for other chipsets,
62 * bridges or combinations thereof.
63 * All comments in this file are by me, regardless if marked TW or not.
64 *
65 */
66
67#if 1
68#define SET_EMI /* 302LV/ELV: Set EMI values */
69#endif
70
71#define COMPAL_HACK /* Needed for Compal 1400x1050 (EMI) */
72#define COMPAQ_HACK /* Needed for Inventec/Compaq 1280x1024 (EMI) */
73#define ASUS_HACK /* Needed for Asus A2H 1024x768 (EMI) */
74
75#include "init301.h"
76
77#ifdef SIS300
78#include "oem300.h"
79#endif
80
81#ifdef SIS315H
82#include "oem310.h"
83#endif
84
85#define SiS_I2CDELAY 1000
86#define SiS_I2CDELAYSHORT 150
87
88static USHORT SiS_GetBIOSLCDResInfo(SiS_Private *SiS_Pr);
Adrian Bunk75c96f82005-05-05 16:16:09 -070089static void SiS_SetCH70xx(SiS_Private *SiS_Pr, USHORT tempbx);
Linus Torvalds1da177e2005-04-16 15:20:36 -070090
91/*********************************************/
92/* HELPER: Lock/Unlock CRT2 */
93/*********************************************/
94
95void
96SiS_UnLockCRT2(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
97{
98 if(HwInfo->jChipType >= SIS_315H)
99 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x2f,0x01);
100 else
101 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x24,0x01);
102}
103
Adrian Bunk75c96f82005-05-05 16:16:09 -0700104static void
Linus Torvalds1da177e2005-04-16 15:20:36 -0700105SiS_LockCRT2(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
106{
107 if(HwInfo->jChipType >= SIS_315H)
108 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x2F,0xFE);
109 else
110 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x24,0xFE);
111}
112
113/*********************************************/
114/* HELPER: Write SR11 */
115/*********************************************/
116
117static void
118SiS_SetRegSR11ANDOR(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo, USHORT DataAND, USHORT DataOR)
119{
120 if(HwInfo->jChipType >= SIS_661) {
121 DataAND &= 0x0f;
122 DataOR &= 0x0f;
123 }
124 SiS_SetRegANDOR(SiS_Pr->SiS_P3c4,0x11,DataAND,DataOR);
125}
126
127/*********************************************/
128/* HELPER: Get Pointer to LCD structure */
129/*********************************************/
130
131#ifdef SIS315H
132static UCHAR *
133GetLCDStructPtr661(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
134{
135 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
136 UCHAR *myptr = NULL;
137 USHORT romindex = 0, reg = 0, idx = 0;
138
139 /* Use the BIOS tables only for LVDS panels; TMDS is unreliable
140 * due to the variaty of panels the BIOS doesn't know about.
141 * Exception: If the BIOS has better knowledge (such as in case
142 * of machines with a 301C and a panel that does not support DDC)
143 * use the BIOS data as well.
144 */
145
146 if((SiS_Pr->SiS_ROMNew) &&
147 ((SiS_Pr->SiS_VBType & VB_SIS301LV302LV) || (!SiS_Pr->PanelSelfDetected))) {
148
149 if(HwInfo->jChipType < SIS_661) reg = 0x3c;
150 else reg = 0x7d;
151
152 idx = (SiS_GetReg(SiS_Pr->SiS_P3d4,reg) & 0x1f) * 26;
153
154 if(idx < (8*26)) {
155 myptr = (UCHAR *)&SiS_LCDStruct661[idx];
156 }
157 romindex = SISGETROMW(0x100);
158 if(romindex) {
159 romindex += idx;
160 myptr = &ROMAddr[romindex];
161 }
162 }
163 return myptr;
164}
165
166static USHORT
167GetLCDStructPtr661_2(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
168{
169 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
170 USHORT romptr = 0;
171
172 /* Use the BIOS tables only for LVDS panels; TMDS is unreliable
173 * due to the variaty of panels the BIOS doesn't know about.
174 * Exception: If the BIOS has better knowledge (such as in case
175 * of machines with a 301C and a panel that does not support DDC)
176 * use the BIOS data as well.
177 */
178
179 if((SiS_Pr->SiS_ROMNew) &&
180 ((SiS_Pr->SiS_VBType & VB_SIS301LV302LV) || (!SiS_Pr->PanelSelfDetected))) {
181 romptr = SISGETROMW(0x102);
182 romptr += ((SiS_GetReg(SiS_Pr->SiS_P3d4,0x36) >> 4) * SiS_Pr->SiS661LCD2TableSize);
183 }
184
185 return(romptr);
186}
187#endif
188
189/*********************************************/
190/* Adjust Rate for CRT2 */
191/*********************************************/
192
193static BOOLEAN
194SiS_AdjustCRT2Rate(SiS_Private *SiS_Pr, USHORT ModeNo, USHORT ModeIdIndex,
195 USHORT RRTI, USHORT *i, PSIS_HW_INFO HwInfo)
196{
197 USHORT checkmask=0,modeid,infoflag;
198
199 modeid = SiS_Pr->SiS_RefIndex[RRTI + (*i)].ModeID;
200
201 if(SiS_Pr->SiS_VBType & VB_SISVB) {
202
203 if(SiS_Pr->SiS_VBInfo & SetCRT2ToRAMDAC) {
204
205 checkmask |= SupportRAMDAC2;
206 if(HwInfo->jChipType >= SIS_315H) {
207 checkmask |= SupportRAMDAC2_135;
208 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) {
209 checkmask |= SupportRAMDAC2_162;
210 if(SiS_Pr->SiS_VBType & VB_SIS301C) {
211 checkmask |= SupportRAMDAC2_202;
212 }
213 }
214 }
215
216 } else if(SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA)) {
217
218 checkmask |= SupportLCD;
219 if(HwInfo->jChipType >= SIS_315H) {
220 if(SiS_Pr->SiS_VBType & VB_SISVB) {
221 if((SiS_Pr->SiS_LCDInfo & DontExpandLCD) && (SiS_Pr->SiS_LCDInfo & LCDPass11)) {
222 if(modeid == 0x2e) checkmask |= Support64048060Hz;
223 }
224 }
225 }
226
227 } else if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) {
228
229 checkmask |= SupportHiVision;
230
231 } else if(SiS_Pr->SiS_VBInfo & (SetCRT2ToYPbPr525750|SetCRT2ToAVIDEO|SetCRT2ToSVIDEO|SetCRT2ToSCART)) {
232
233 checkmask |= SupportTV;
234 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) {
235 checkmask |= SupportTV1024;
236 if(SiS_Pr->SiS_VBInfo & SetCRT2ToYPbPr525750) {
237 if(SiS_Pr->SiS_TVMode & TVSetYPbPr750p) {
238 checkmask |= SupportYPbPr750p;
239 }
240 }
241 }
242
243 }
244
245 } else { /* LVDS */
246
247 if(SiS_Pr->SiS_IF_DEF_CH70xx != 0) {
248 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) {
249 checkmask |= SupportCHTV;
250 }
251 }
252
253 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) {
254 checkmask |= SupportLCD;
255 }
256
257 }
258
259 /* Look backwards in table for matching CRT2 mode */
260 for(; SiS_Pr->SiS_RefIndex[RRTI + (*i)].ModeID == modeid; (*i)--) {
261 infoflag = SiS_Pr->SiS_RefIndex[RRTI + (*i)].Ext_InfoFlag;
262 if(infoflag & checkmask) return TRUE;
263 if((*i) == 0) break;
264 }
265
266 /* Look through the whole mode-section of the table from the beginning
267 * for a matching CRT2 mode if no mode was found yet.
268 */
269 for((*i) = 0; ; (*i)++) {
270 if(SiS_Pr->SiS_RefIndex[RRTI + (*i)].ModeID != modeid) break;
271 infoflag = SiS_Pr->SiS_RefIndex[RRTI + (*i)].Ext_InfoFlag;
272 if(infoflag & checkmask) return TRUE;
273 }
274 return FALSE;
275}
276
277/*********************************************/
278/* Get rate index */
279/*********************************************/
280
281USHORT
282SiS_GetRatePtr(SiS_Private *SiS_Pr, USHORT ModeNo, USHORT ModeIdIndex,
283 PSIS_HW_INFO HwInfo)
284{
285 SHORT LCDRefreshIndex[] = { 0x00, 0x00, 0x01, 0x01,
286 0x01, 0x01, 0x01, 0x01,
287 0x01, 0x01, 0x01, 0x01,
288 0x01, 0x01, 0x01, 0x01,
289 0x00, 0x00, 0x00, 0x00 };
290 USHORT RRTI,i,backup_i;
291 USHORT modeflag,index,temp,backupindex;
292
293 /* Do NOT check for UseCustomMode here, will skrew up FIFO */
294 if(ModeNo == 0xfe) return 0;
295
296 if(ModeNo <= 0x13) {
297 modeflag = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_ModeFlag;
298 } else {
299 modeflag = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_ModeFlag;
300 }
301
302 if(SiS_Pr->SiS_IF_DEF_CH70xx != 0) {
303 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) {
304 if(modeflag & HalfDCLK) return 0;
305 }
306 }
307
308 if(ModeNo < 0x14) return 0xFFFF;
309
310 index = (SiS_GetReg(SiS_Pr->SiS_P3d4,0x33) >> SiS_Pr->SiS_SelectCRT2Rate) & 0x0F;
311 backupindex = index;
312
313 if(index > 0) index--;
314
315 if(SiS_Pr->SiS_SetFlag & ProgrammingCRT2) {
316 if(SiS_Pr->SiS_VBType & VB_SISVB) {
317 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) {
318 if(SiS_Pr->SiS_VBType & VB_NoLCD) index = 0;
319 else if(SiS_Pr->SiS_LCDInfo & DontExpandLCD) index = backupindex = 0;
320 }
321 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA)) {
322 if(!(SiS_Pr->SiS_VBType & VB_NoLCD)) {
323 temp = LCDRefreshIndex[SiS_GetBIOSLCDResInfo(SiS_Pr)];
324 if(index > temp) index = temp;
325 }
326 }
327 } else {
328 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA)) index = 0;
329 if(SiS_Pr->SiS_IF_DEF_CH70xx != 0) {
330 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) index = 0;
331 }
332 }
333 }
334
335 RRTI = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].REFindex;
336 ModeNo = SiS_Pr->SiS_RefIndex[RRTI].ModeID;
337
338 if(HwInfo->jChipType >= SIS_315H) {
339 if(!(SiS_Pr->SiS_VBInfo & DriverMode)) {
340 if( (SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_VESAID == 0x105) ||
341 (SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_VESAID == 0x107) ) {
342 if(backupindex <= 1) RRTI++;
343 }
344 }
345 }
346
347 i = 0;
348 do {
349 if(SiS_Pr->SiS_RefIndex[RRTI + i].ModeID != ModeNo) break;
350 temp = SiS_Pr->SiS_RefIndex[RRTI + i].Ext_InfoFlag;
351 temp &= ModeTypeMask;
352 if(temp < SiS_Pr->SiS_ModeType) break;
353 i++;
354 index--;
355 } while(index != 0xFFFF);
356
357 if(!(SiS_Pr->SiS_VBInfo & SetCRT2ToRAMDAC)) {
358 if(SiS_Pr->SiS_VBInfo & SetInSlaveMode) {
359 temp = SiS_Pr->SiS_RefIndex[RRTI + i - 1].Ext_InfoFlag;
360 if(temp & InterlaceMode) i++;
361 }
362 }
363
364 i--;
365
366 if((SiS_Pr->SiS_SetFlag & ProgrammingCRT2) && (!(SiS_Pr->SiS_VBInfo & DisableCRT2Display))) {
367 backup_i = i;
368 if(!(SiS_AdjustCRT2Rate(SiS_Pr, ModeNo, ModeIdIndex, RRTI, &i, HwInfo))) {
369 i = backup_i;
370 }
371 }
372
373 return(RRTI + i);
374}
375
376/*********************************************/
377/* STORE CRT2 INFO in CR34 */
378/*********************************************/
379
380static void
381SiS_SaveCRT2Info(SiS_Private *SiS_Pr, USHORT ModeNo)
382{
383 USHORT temp1,temp2;
384
385 /* Store CRT1 ModeNo in CR34 */
386 SiS_SetReg(SiS_Pr->SiS_P3d4,0x34,ModeNo);
387 temp1 = (SiS_Pr->SiS_VBInfo & SetInSlaveMode) >> 8;
388 temp2 = ~(SetInSlaveMode >> 8);
389 SiS_SetRegANDOR(SiS_Pr->SiS_P3d4,0x31,temp2,temp1);
390}
391
392/*********************************************/
393/* HELPER: GET SOME DATA FROM BIOS ROM */
394/*********************************************/
395
396#ifdef SIS300
397static BOOLEAN
398SiS_CR36BIOSWord23b(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
399{
400 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
401 USHORT temp,temp1;
402
403 if(SiS_Pr->SiS_UseROM) {
404 if((ROMAddr[0x233] == 0x12) && (ROMAddr[0x234] == 0x34)) {
405 temp = 1 << ((SiS_GetReg(SiS_Pr->SiS_P3d4,0x36) >> 4) & 0x0f);
406 temp1 = SISGETROMW(0x23b);
407 if(temp1 & temp) return TRUE;
408 }
409 }
410 return FALSE;
411}
412
413static BOOLEAN
414SiS_CR36BIOSWord23d(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
415{
416 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
417 USHORT temp,temp1;
418
419 if(SiS_Pr->SiS_UseROM) {
420 if((ROMAddr[0x233] == 0x12) && (ROMAddr[0x234] == 0x34)) {
421 temp = 1 << ((SiS_GetReg(SiS_Pr->SiS_P3d4,0x36) >> 4) & 0x0f);
422 temp1 = SISGETROMW(0x23d);
423 if(temp1 & temp) return TRUE;
424 }
425 }
426 return FALSE;
427}
428#endif
429
430/*********************************************/
431/* HELPER: DELAY FUNCTIONS */
432/*********************************************/
433
434void
435SiS_DDC2Delay(SiS_Private *SiS_Pr, USHORT delaytime)
436{
437 USHORT i, j;
438
439 for(i=0; i<delaytime; i++) {
440 j += SiS_GetReg(SiS_Pr->SiS_P3c4,0x05);
441 }
442}
443
444#if defined(SIS300) || defined(SIS315H)
445static void
446SiS_GenericDelay(SiS_Private *SiS_Pr, USHORT delay)
447{
448 USHORT temp,flag;
449
450 flag = SiS_GetRegByte(0x61) & 0x10;
451
452 while(delay) {
453 temp = SiS_GetRegByte(0x61) & 0x10;
454 if(temp == flag) continue;
455 flag = temp;
456 delay--;
457 }
458}
459#endif
460
461#ifdef SIS315H
462static void
463SiS_LongDelay(SiS_Private *SiS_Pr, USHORT delay)
464{
465 while(delay--) {
466 SiS_GenericDelay(SiS_Pr,0x19df);
467 }
468}
469#endif
470
471#if defined(SIS300) || defined(SIS315H)
472static void
473SiS_ShortDelay(SiS_Private *SiS_Pr, USHORT delay)
474{
475 while(delay--) {
476 SiS_GenericDelay(SiS_Pr,0x42);
477 }
478}
479#endif
480
481static void
482SiS_PanelDelay(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo, USHORT DelayTime)
483{
484#if defined(SIS300) || defined(SIS315H)
485 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
486 USHORT PanelID, DelayIndex, Delay=0;
487#endif
488
489 if(HwInfo->jChipType < SIS_315H) {
490
491#ifdef SIS300
492
493 PanelID = SiS_GetReg(SiS_Pr->SiS_P3d4,0x36);
494 if(SiS_Pr->SiS_VBType & VB_SISVB) {
495 if(SiS_Pr->SiS_VBType & VB_SIS301) PanelID &= 0xf7;
496 if(!(SiS_GetReg(SiS_Pr->SiS_P3c4,0x18) & 0x10)) PanelID = 0x12;
497 }
498 DelayIndex = PanelID >> 4;
499 if((DelayTime >= 2) && ((PanelID & 0x0f) == 1)) {
500 Delay = 3;
501 } else {
502 if(DelayTime >= 2) DelayTime -= 2;
503 if(!(DelayTime & 0x01)) {
504 Delay = SiS_Pr->SiS_PanelDelayTbl[DelayIndex].timer[0];
505 } else {
506 Delay = SiS_Pr->SiS_PanelDelayTbl[DelayIndex].timer[1];
507 }
508 if(SiS_Pr->SiS_UseROM) {
509 if(ROMAddr[0x220] & 0x40) {
510 if(!(DelayTime & 0x01)) Delay = (USHORT)ROMAddr[0x225];
511 else Delay = (USHORT)ROMAddr[0x226];
512 }
513 }
514 }
515 SiS_ShortDelay(SiS_Pr, Delay);
516
517#endif /* SIS300 */
518
519 } else {
520
521#ifdef SIS315H
522
523 if((HwInfo->jChipType >= SIS_661) ||
524 (HwInfo->jChipType <= SIS_315PRO) ||
525 (HwInfo->jChipType == SIS_330) ||
526 (SiS_Pr->SiS_ROMNew)) {
527
528 if(!(DelayTime & 0x01)) {
529 SiS_DDC2Delay(SiS_Pr, 0x1000);
530 } else {
531 SiS_DDC2Delay(SiS_Pr, 0x4000);
532 }
533
534 } else if((SiS_Pr->SiS_IF_DEF_LVDS == 1) /* ||
535 (SiS_Pr->SiS_CustomT == CUT_COMPAQ1280) ||
536 (SiS_Pr->SiS_CustomT == CUT_CLEVO1400) */ ) { /* 315 series, LVDS; Special */
537
538 if(SiS_Pr->SiS_IF_DEF_CH70xx == 0) {
539 PanelID = SiS_GetReg(SiS_Pr->SiS_P3d4,0x36);
540 if(SiS_Pr->SiS_CustomT == CUT_CLEVO1400) {
541 if(!(SiS_GetReg(SiS_Pr->SiS_P3c4,0x1b) & 0x10)) PanelID = 0x12;
542 }
543 if(SiS_Pr->SiS_CustomT == CUT_COMPAQ1280) {
544 DelayIndex = PanelID & 0x0f;
545 } else {
546 DelayIndex = PanelID >> 4;
547 }
548 if((DelayTime >= 2) && ((PanelID & 0x0f) == 1)) {
549 Delay = 3;
550 } else {
551 if(DelayTime >= 2) DelayTime -= 2;
552 if(!(DelayTime & 0x01)) {
553 Delay = SiS_Pr->SiS_PanelDelayTblLVDS[DelayIndex].timer[0];
554 } else {
555 Delay = SiS_Pr->SiS_PanelDelayTblLVDS[DelayIndex].timer[1];
556 }
557 if((SiS_Pr->SiS_UseROM) && (!(SiS_Pr->SiS_ROMNew))) {
558 if(ROMAddr[0x13c] & 0x40) {
559 if(!(DelayTime & 0x01)) {
560 Delay = (USHORT)ROMAddr[0x17e];
561 } else {
562 Delay = (USHORT)ROMAddr[0x17f];
563 }
564 }
565 }
566 }
567 SiS_ShortDelay(SiS_Pr, Delay);
568 }
569
570 } else if(SiS_Pr->SiS_VBType & VB_SISVB) { /* 315 series, all bridges */
571
572 DelayIndex = SiS_GetReg(SiS_Pr->SiS_P3d4,0x36) >> 4;
573 if(!(DelayTime & 0x01)) {
574 Delay = SiS_Pr->SiS_PanelDelayTbl[DelayIndex].timer[0];
575 } else {
576 Delay = SiS_Pr->SiS_PanelDelayTbl[DelayIndex].timer[1];
577 }
578 Delay <<= 8;
579 SiS_DDC2Delay(SiS_Pr, Delay);
580
581 }
582
583#endif /* SIS315H */
584
585 }
586}
587
588#ifdef SIS315H
589static void
590SiS_PanelDelayLoop(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo,
591 USHORT DelayTime, USHORT DelayLoop)
592{
593 int i;
594 for(i=0; i<DelayLoop; i++) {
595 SiS_PanelDelay(SiS_Pr, HwInfo, DelayTime);
596 }
597}
598#endif
599
600/*********************************************/
601/* HELPER: WAIT-FOR-RETRACE FUNCTIONS */
602/*********************************************/
603
604void
605SiS_WaitRetrace1(SiS_Private *SiS_Pr)
606{
607 USHORT watchdog;
608
609 if(SiS_GetReg(SiS_Pr->SiS_P3c4,0x1f) & 0xc0) return;
610 if(!(SiS_GetReg(SiS_Pr->SiS_P3d4,0x17) & 0x80)) return;
611
612 watchdog = 65535;
613 while((SiS_GetRegByte(SiS_Pr->SiS_P3da) & 0x08) && --watchdog);
614 watchdog = 65535;
615 while((!(SiS_GetRegByte(SiS_Pr->SiS_P3da) & 0x08)) && --watchdog);
616}
617
618#if defined(SIS300) || defined(SIS315H)
619static void
620SiS_WaitRetrace2(SiS_Private *SiS_Pr, USHORT reg)
621{
622 USHORT watchdog;
623
624 watchdog = 65535;
625 while((SiS_GetReg(SiS_Pr->SiS_Part1Port,reg) & 0x02) && --watchdog);
626 watchdog = 65535;
627 while((!(SiS_GetReg(SiS_Pr->SiS_Part1Port,reg) & 0x02)) && --watchdog);
628}
629#endif
630
631static void
632SiS_WaitVBRetrace(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
633{
634 if(HwInfo->jChipType < SIS_315H) {
635#ifdef SIS300
636 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) {
637 if(!(SiS_GetReg(SiS_Pr->SiS_Part1Port,0x00) & 0x20)) return;
638 }
639 if(!(SiS_GetReg(SiS_Pr->SiS_Part1Port,0x00) & 0x80)) {
640 SiS_WaitRetrace1(SiS_Pr);
641 } else {
642 SiS_WaitRetrace2(SiS_Pr, 0x25);
643 }
644#endif
645 } else {
646#ifdef SIS315H
647 if(!(SiS_GetReg(SiS_Pr->SiS_Part1Port,0x00) & 0x40)) {
648 SiS_WaitRetrace1(SiS_Pr);
649 } else {
650 SiS_WaitRetrace2(SiS_Pr, 0x30);
651 }
652#endif
653 }
654}
655
656static void
657SiS_VBWait(SiS_Private *SiS_Pr)
658{
659 USHORT tempal,temp,i,j;
660
661 temp = 0;
662 for(i=0; i<3; i++) {
663 for(j=0; j<100; j++) {
664 tempal = SiS_GetRegByte(SiS_Pr->SiS_P3da);
665 if(temp & 0x01) {
666 if((tempal & 0x08)) continue;
667 else break;
668 } else {
669 if(!(tempal & 0x08)) continue;
670 else break;
671 }
672 }
673 temp ^= 0x01;
674 }
675}
676
677static void
678SiS_VBLongWait(SiS_Private *SiS_Pr)
679{
680 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) {
681 SiS_VBWait(SiS_Pr);
682 } else {
683 SiS_WaitRetrace1(SiS_Pr);
684 }
685}
686
687/*********************************************/
688/* HELPER: MISC */
689/*********************************************/
690
691#ifdef SIS300
692static BOOLEAN
693SiS_Is301B(SiS_Private *SiS_Pr)
694{
695 if(SiS_GetReg(SiS_Pr->SiS_Part4Port,0x01) >= 0xb0) return TRUE;
696 return FALSE;
697}
698#endif
699
700static BOOLEAN
701SiS_CRT2IsLCD(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
702{
703 USHORT flag;
704
705 if(HwInfo->jChipType == SIS_730) {
706 flag = SiS_GetReg(SiS_Pr->SiS_P3c4,0x13);
707 if(flag & 0x20) return TRUE;
708 }
709 flag = SiS_GetReg(SiS_Pr->SiS_P3d4,0x30);
710 if(flag & 0x20) return TRUE;
711 return FALSE;
712}
713
714BOOLEAN
715SiS_IsDualEdge(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
716{
717#ifdef SIS315H
718 USHORT flag;
719
720 if(HwInfo->jChipType >= SIS_315H) {
721 if((HwInfo->jChipType != SIS_650) || (SiS_GetReg(SiS_Pr->SiS_P3d4,0x5f) & 0xf0)) {
722 flag = SiS_GetReg(SiS_Pr->SiS_P3d4,0x38);
723 if(flag & EnableDualEdge) return TRUE;
724 }
725 }
726#endif
727 return FALSE;
728}
729
730BOOLEAN
731SiS_IsVAMode(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
732{
733#ifdef SIS315H
734 USHORT flag;
735
736 if(HwInfo->jChipType >= SIS_315H) {
737 flag = SiS_GetReg(SiS_Pr->SiS_P3d4,0x38);
738 if((flag & EnableDualEdge) && (flag & SetToLCDA)) return TRUE;
739 }
740#endif
741 return FALSE;
742}
743
744#ifdef SIS315H
745static BOOLEAN
746SiS_IsVAorLCD(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
747{
748 if(SiS_IsVAMode(SiS_Pr,HwInfo)) return TRUE;
749 if(SiS_CRT2IsLCD(SiS_Pr, HwInfo)) return TRUE;
750 return FALSE;
751}
752#endif
753
754static BOOLEAN
755SiS_IsDualLink(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
756{
757#ifdef SIS315H
758 if(HwInfo->jChipType >= SIS_315H) {
759 if((SiS_CRT2IsLCD(SiS_Pr, HwInfo)) ||
760 (SiS_IsVAMode(SiS_Pr, HwInfo))) {
761 if(SiS_Pr->SiS_LCDInfo & LCDDualLink) return TRUE;
762 }
763 }
764#endif
765 return FALSE;
766}
767
768#ifdef SIS315H
769static BOOLEAN
770SiS_TVEnabled(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
771{
772 if((SiS_GetReg(SiS_Pr->SiS_Part2Port,0x00) & 0x0f) != 0x0c) return TRUE;
773 if(SiS_Pr->SiS_VBType & (VB_SIS301C | VB_SIS301LV302LV)) {
774 if(SiS_GetReg(SiS_Pr->SiS_Part2Port,0x4d) & 0x10) return TRUE;
775 }
776 return FALSE;
777}
778#endif
779
780#ifdef SIS315H
781static BOOLEAN
782SiS_LCDAEnabled(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
783{
784 if(SiS_GetReg(SiS_Pr->SiS_Part1Port,0x13) & 0x04) return TRUE;
785 return FALSE;
786}
787#endif
788
789#ifdef SIS315H
790static BOOLEAN
791SiS_WeHaveBacklightCtrl(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
792{
793 if((HwInfo->jChipType >= SIS_315H) && (HwInfo->jChipType < SIS_661)) {
794 if(SiS_GetReg(SiS_Pr->SiS_P3d4,0x79) & 0x10) return TRUE;
795 }
796 return FALSE;
797}
798#endif
799
800#ifdef SIS315H
801static BOOLEAN
802SiS_IsNotM650orLater(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
803{
804 USHORT flag;
805
806 if(HwInfo->jChipType == SIS_650) {
807 flag = SiS_GetReg(SiS_Pr->SiS_P3d4,0x5f);
808 flag &= 0xF0;
809 /* Check for revision != A0 only */
810 if((flag == 0xe0) || (flag == 0xc0) ||
811 (flag == 0xb0) || (flag == 0x90)) return FALSE;
812 } else if(HwInfo->jChipType >= SIS_661) return FALSE;
813 return TRUE;
814}
815#endif
816
817#ifdef SIS315H
818static BOOLEAN
819SiS_IsYPbPr(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
820{
821 USHORT flag;
822
823 if(HwInfo->jChipType >= SIS_315H) {
824 flag = SiS_GetReg(SiS_Pr->SiS_P3d4,0x38);
825 if(flag & EnableCHYPbPr) return TRUE; /* = YPrPb = 0x08 */
826 }
827 return FALSE;
828}
829#endif
830
831#ifdef SIS315H
832static BOOLEAN
833SiS_IsChScart(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
834{
835 USHORT flag;
836
837 if(HwInfo->jChipType >= SIS_315H) {
838 flag = SiS_GetReg(SiS_Pr->SiS_P3d4,0x38);
839 if(flag & EnableCHScart) return TRUE; /* = Scart = 0x04 */
840 }
841 return FALSE;
842}
843#endif
844
845#ifdef SIS315H
846static BOOLEAN
847SiS_IsTVOrYPbPrOrScart(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
848{
849 USHORT flag;
850
851 if(HwInfo->jChipType >= SIS_315H) {
852 flag = SiS_GetReg(SiS_Pr->SiS_P3d4,0x30);
853 if(flag & SetCRT2ToTV) return TRUE;
854 flag = SiS_GetReg(SiS_Pr->SiS_P3d4,0x38);
855 if(flag & EnableCHYPbPr) return TRUE; /* = YPrPb = 0x08 */
856 if(flag & EnableCHScart) return TRUE; /* = Scart = 0x04 - TW */
857 } else {
858 flag = SiS_GetReg(SiS_Pr->SiS_P3d4,0x30);
859 if(flag & SetCRT2ToTV) return TRUE;
860 }
861 return FALSE;
862}
863#endif
864
865#ifdef SIS315H
866static BOOLEAN
867SiS_IsLCDOrLCDA(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
868{
869 USHORT flag;
870
871 if(HwInfo->jChipType >= SIS_315H) {
872 flag = SiS_GetReg(SiS_Pr->SiS_P3d4,0x30);
873 if(flag & SetCRT2ToLCD) return TRUE;
874 flag = SiS_GetReg(SiS_Pr->SiS_P3d4,0x38);
875 if(flag & SetToLCDA) return TRUE;
876 } else {
877 flag = SiS_GetReg(SiS_Pr->SiS_P3d4,0x30);
878 if(flag & SetCRT2ToLCD) return TRUE;
879 }
880 return FALSE;
881}
882#endif
883
884static BOOLEAN
885SiS_BridgeIsOn(SiS_Private *SiS_Pr)
886{
887 USHORT flag;
888
889 if(SiS_Pr->SiS_IF_DEF_LVDS == 1) {
890 return TRUE;
891 } else if(SiS_Pr->SiS_VBType & VB_SISVB) {
892 flag = SiS_GetReg(SiS_Pr->SiS_Part4Port,0x00);
893 if((flag == 1) || (flag == 2)) return TRUE;
894 }
895 return FALSE;
896}
897
898static BOOLEAN
899SiS_BridgeIsEnabled(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
900{
901 USHORT flag;
902
903 if(SiS_BridgeIsOn(SiS_Pr)) {
904 flag = SiS_GetReg(SiS_Pr->SiS_Part1Port,0x00);
905 if(HwInfo->jChipType < SIS_315H) {
906 flag &= 0xa0;
907 if((flag == 0x80) || (flag == 0x20)) return TRUE;
908 } else {
909 flag &= 0x50;
910 if((flag == 0x40) || (flag == 0x10)) return TRUE;
911 }
912 }
913 return FALSE;
914}
915
916static BOOLEAN
917SiS_BridgeInSlavemode(SiS_Private *SiS_Pr)
918{
919 USHORT flag1;
920
921 flag1 = SiS_GetReg(SiS_Pr->SiS_P3d4,0x31);
922 if(flag1 & (SetInSlaveMode >> 8)) return TRUE;
923 return FALSE;
924}
925
926/*********************************************/
927/* GET VIDEO BRIDGE CONFIG INFO */
928/*********************************************/
929
930/* Setup general purpose IO for Chrontel communication */
931void
932SiS_SetChrontelGPIO(SiS_Private *SiS_Pr, USHORT myvbinfo)
933{
934 unsigned long acpibase;
935 unsigned short temp;
936
937 if(!(SiS_Pr->SiS_ChSW)) return;
938
939#ifdef LINUX_KERNEL
940 SiS_SetRegLong(0xcf8,0x80000874); /* get ACPI base */
941 acpibase = SiS_GetRegLong(0xcfc);
942#else
943 acpibase = pciReadLong(0x00000800, 0x74);
944#endif
945 acpibase &= 0xFFFF;
946 temp = SiS_GetRegShort((USHORT)(acpibase + 0x3c)); /* ACPI register 0x3c: GP Event 1 I/O mode select */
947 temp &= 0xFEFF;
948 SiS_SetRegShort((USHORT)(acpibase + 0x3c), temp);
949 temp = SiS_GetRegShort((USHORT)(acpibase + 0x3c));
950 temp = SiS_GetRegShort((USHORT)(acpibase + 0x3a)); /* ACPI register 0x3a: GP Pin Level (low/high) */
951 temp &= 0xFEFF;
952 if(!(myvbinfo & SetCRT2ToTV)) temp |= 0x0100;
953 SiS_SetRegShort((USHORT)(acpibase + 0x3a), temp);
954 temp = SiS_GetRegShort((USHORT)(acpibase + 0x3a));
955}
956
957void
958SiS_GetVBInfo(SiS_Private *SiS_Pr, USHORT ModeNo, USHORT ModeIdIndex,
959 PSIS_HW_INFO HwInfo, int checkcrt2mode)
960{
961 USHORT tempax,tempbx,temp;
962 USHORT modeflag, resinfo=0;
963
964 if(ModeNo <= 0x13) {
965 modeflag = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_ModeFlag;
966 } else if(SiS_Pr->UseCustomMode) {
967 modeflag = SiS_Pr->CModeFlag;
968 } else {
969 modeflag = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_ModeFlag;
970 resinfo = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_RESINFO;
971 }
972
973 SiS_Pr->SiS_SetFlag = 0;
974
975 SiS_Pr->SiS_ModeType = modeflag & ModeTypeMask;
976
977 tempbx = 0;
978 if(SiS_BridgeIsOn(SiS_Pr)) {
979 temp = SiS_GetReg(SiS_Pr->SiS_P3d4,0x30);
980#if 0
981 if(HwInfo->jChipType < SIS_661) {
982 /* NO - YPbPr not set yet ! */
983 if(SiS_Pr->SiS_YPbPr & <all ypbpr except 525i>) {
984 temp &= (SetCRT2ToHiVision | SwitchCRT2 | SetSimuScanMode); /* 0x83 */
985 temp |= SetCRT2ToHiVision; /* 0x80 */
986 }
987 if(SiS_Pr->SiS_YPbPr & <ypbpr525i>) {
988 temp &= (SetCRT2ToHiVision | SwitchCRT2 | SetSimuScanMode); /* 0x83 */
989 temp |= SetCRT2ToSVIDEO; /* 0x08 */
990 }
991 }
992#endif
993 tempbx |= temp;
994 tempax = SiS_GetReg(SiS_Pr->SiS_P3d4,0x31) << 8;
995 tempax &= (DriverMode | LoadDACFlag | SetNotSimuMode | SetPALTV);
996 tempbx |= tempax;
997
998#ifdef SIS315H
999 if(HwInfo->jChipType >= SIS_315H) {
1000 if(SiS_Pr->SiS_VBType & VB_SISLCDA) {
1001 if(ModeNo == 0x03) {
1002 /* Mode 0x03 is never in driver mode */
1003 SiS_SetRegAND(SiS_Pr->SiS_P3d4,0x31,0xbf);
1004 }
1005 if(!(SiS_GetReg(SiS_Pr->SiS_P3d4,0x31) & (DriverMode >> 8))) {
1006 /* Reset LCDA setting if not driver mode */
1007 SiS_SetRegAND(SiS_Pr->SiS_P3d4,0x38,0xfc);
1008 }
1009 if(IS_SIS650) {
1010 if(SiS_Pr->SiS_UseLCDA) {
1011 if(SiS_GetReg(SiS_Pr->SiS_P3d4,0x5f) & 0xF0) {
1012 if((ModeNo <= 0x13) || (!(SiS_GetReg(SiS_Pr->SiS_P3d4,0x31) & (DriverMode >> 8)))) {
1013 SiS_SetRegOR(SiS_Pr->SiS_P3d4,0x38,(EnableDualEdge | SetToLCDA));
1014 }
1015 }
1016 }
1017 }
1018 temp = SiS_GetReg(SiS_Pr->SiS_P3d4,0x38);
1019 if((temp & (EnableDualEdge | SetToLCDA)) == (EnableDualEdge | SetToLCDA)) {
1020 tempbx |= SetCRT2ToLCDA;
1021 }
1022 }
1023
1024 if(SiS_Pr->SiS_VBType & (VB_SIS301LV|VB_SIS302LV|VB_SIS302ELV)) {
1025 tempbx &= ~(SetCRT2ToRAMDAC);
1026 }
1027
1028 if(HwInfo->jChipType >= SIS_661) {
1029 tempbx &= ~(SetCRT2ToYPbPr525750 | SetCRT2ToHiVision);
1030 temp = SiS_GetReg(SiS_Pr->SiS_P3d4,0x38);
1031 if(SiS_Pr->SiS_VBType & VB_SISYPBPR) {
1032 if(temp & 0x04) {
1033 temp = SiS_GetReg(SiS_Pr->SiS_P3d4,0x35) & 0xe0;
1034 if(temp == 0x60) tempbx |= SetCRT2ToHiVision;
1035 else tempbx |= SetCRT2ToYPbPr525750;
1036 }
1037 } else if(SiS_Pr->SiS_VBType & VB_SISHIVISION) {
1038 if(temp & 0x04) {
1039 temp = SiS_GetReg(SiS_Pr->SiS_P3d4,0x35) & 0xe0;
1040 if(temp == 0x60) tempbx |= SetCRT2ToHiVision;
1041 }
1042 }
1043 }
1044
1045 if(SiS_Pr->SiS_IF_DEF_LVDS == 1) {
1046 temp = SiS_GetReg(SiS_Pr->SiS_P3d4,0x38);
1047 if(temp & SetToLCDA) {
1048 tempbx |= SetCRT2ToLCDA;
1049 }
1050 if(SiS_Pr->SiS_IF_DEF_CH70xx != 0) {
1051 if(temp & EnableCHYPbPr) {
1052 tempbx |= SetCRT2ToCHYPbPr;
1053 }
1054 }
1055 }
1056 }
1057
1058#endif /* SIS315H */
1059
1060 if(SiS_Pr->SiS_VBType & VB_SISVB) {
1061 temp = SetCRT2ToSVIDEO |
1062 SetCRT2ToAVIDEO |
1063 SetCRT2ToSCART |
1064 SetCRT2ToLCDA |
1065 SetCRT2ToLCD |
1066 SetCRT2ToRAMDAC |
1067 SetCRT2ToHiVision |
1068 SetCRT2ToYPbPr525750;
1069 } else {
1070 if(HwInfo->jChipType >= SIS_315H) {
1071 if(SiS_Pr->SiS_IF_DEF_CH70xx != 0) {
1072 temp = SetCRT2ToAVIDEO |
1073 SetCRT2ToSVIDEO |
1074 SetCRT2ToSCART |
1075 SetCRT2ToLCDA |
1076 SetCRT2ToLCD |
1077 SetCRT2ToCHYPbPr;
1078 } else {
1079 temp = SetCRT2ToLCDA |
1080 SetCRT2ToLCD;
1081 }
1082 } else {
1083 if(SiS_Pr->SiS_IF_DEF_CH70xx != 0) {
1084 temp = SetCRT2ToTV | SetCRT2ToLCD;
1085 } else {
1086 temp = SetCRT2ToLCD;
1087 }
1088 }
1089 }
1090
1091 if(!(tempbx & temp)) {
1092 tempax = DisableCRT2Display;
1093 tempbx = 0;
1094 }
1095
1096 if(SiS_Pr->SiS_VBType & VB_SISVB) {
1097 USHORT clearmask = ( DriverMode |
1098 DisableCRT2Display |
1099 LoadDACFlag |
1100 SetNotSimuMode |
1101 SetInSlaveMode |
1102 SetPALTV |
1103 SwitchCRT2 |
1104 SetSimuScanMode );
1105 if(tempbx & SetCRT2ToLCDA) tempbx &= (clearmask | SetCRT2ToLCDA);
1106 if(tempbx & SetCRT2ToRAMDAC) tempbx &= (clearmask | SetCRT2ToRAMDAC);
1107 if(tempbx & SetCRT2ToLCD) tempbx &= (clearmask | SetCRT2ToLCD);
1108 if(tempbx & SetCRT2ToSCART) tempbx &= (clearmask | SetCRT2ToSCART);
1109 if(tempbx & SetCRT2ToHiVision) tempbx &= (clearmask | SetCRT2ToHiVision);
1110 if(tempbx & SetCRT2ToYPbPr525750) tempbx &= (clearmask | SetCRT2ToYPbPr525750);
1111 } else {
1112 if(HwInfo->jChipType >= SIS_315H) {
1113 if(tempbx & SetCRT2ToLCDA) {
1114 tempbx &= (0xFF00|SwitchCRT2|SetSimuScanMode);
1115 }
1116 }
1117 if(SiS_Pr->SiS_IF_DEF_CH70xx != 0) {
1118 if(tempbx & SetCRT2ToTV) {
1119 tempbx &= (0xFF00|SetCRT2ToTV|SwitchCRT2|SetSimuScanMode);
1120 }
1121 }
1122 if(tempbx & SetCRT2ToLCD) {
1123 tempbx &= (0xFF00|SetCRT2ToLCD|SwitchCRT2|SetSimuScanMode);
1124 }
1125 if(HwInfo->jChipType >= SIS_315H) {
1126 if(tempbx & SetCRT2ToLCDA) {
1127 tempbx |= SetCRT2ToLCD;
1128 }
1129 }
1130 }
1131
1132 if(tempax & DisableCRT2Display) {
1133 if(!(tempbx & (SwitchCRT2 | SetSimuScanMode))) {
1134 tempbx = SetSimuScanMode | DisableCRT2Display;
1135 }
1136 }
1137
1138 if(!(tempbx & DriverMode)) tempbx |= SetSimuScanMode;
1139
1140 /* LVDS/CHRONTEL (LCD/TV) and 301BDH (LCD) can only be slave in 8bpp modes */
1141 if(SiS_Pr->SiS_ModeType <= ModeVGA) {
1142 if( (SiS_Pr->SiS_IF_DEF_LVDS == 1) ||
1143 ((SiS_Pr->SiS_VBType & VB_NoLCD) && (tempbx & SetCRT2ToLCD)) ) {
1144 modeflag &= (~CRT2Mode);
1145 }
1146 }
1147
1148 if(!(tempbx & SetSimuScanMode)) {
1149 if(tempbx & SwitchCRT2) {
1150 if((!(modeflag & CRT2Mode)) && (checkcrt2mode)) {
1151 if( (HwInfo->jChipType >= SIS_315H) &&
1152 (SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) ) {
1153 if(resinfo != SIS_RI_1600x1200) {
1154 tempbx |= SetSimuScanMode;
1155 }
1156 } else {
1157 tempbx |= SetSimuScanMode;
1158 }
1159 }
1160 } else {
1161 if(SiS_BridgeIsEnabled(SiS_Pr,HwInfo)) {
1162 if(!(tempbx & DriverMode)) {
1163 if(SiS_BridgeInSlavemode(SiS_Pr)) {
1164 tempbx |= SetSimuScanMode;
1165 }
1166 }
1167 }
1168 }
1169 }
1170
1171 if(!(tempbx & DisableCRT2Display)) {
1172 if(tempbx & DriverMode) {
1173 if(tempbx & SetSimuScanMode) {
1174 if((!(modeflag & CRT2Mode)) && (checkcrt2mode)) {
1175 if( (HwInfo->jChipType >= SIS_315H) &&
1176 (SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) ) {
1177 if(resinfo != SIS_RI_1600x1200) {
1178 tempbx |= SetInSlaveMode;
1179 }
1180 } else {
1181 tempbx |= SetInSlaveMode;
1182 }
1183 }
1184 }
1185 } else {
1186 tempbx |= SetInSlaveMode;
1187 }
1188 }
1189
1190 }
1191
1192 SiS_Pr->SiS_VBInfo = tempbx;
1193
1194 if(HwInfo->jChipType == SIS_630) {
1195 SiS_SetChrontelGPIO(SiS_Pr, SiS_Pr->SiS_VBInfo);
1196 }
1197
1198#ifdef TWDEBUG
1199#ifdef LINUX_KERNEL
1200 printk(KERN_DEBUG "sisfb: (VBInfo= 0x%04x, SetFlag=0x%04x)\n",
1201 SiS_Pr->SiS_VBInfo, SiS_Pr->SiS_SetFlag);
1202#endif
1203#ifdef LINUX_XF86
1204 xf86DrvMsgVerb(0, X_PROBED, 3, "(init301: VBInfo=0x%04x, SetFlag=0x%04x)\n",
1205 SiS_Pr->SiS_VBInfo, SiS_Pr->SiS_SetFlag);
1206#endif
1207#endif
1208}
1209
1210/*********************************************/
1211/* DETERMINE YPbPr MODE */
1212/*********************************************/
1213
1214void
1215SiS_SetYPbPr(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
1216{
1217
1218 UCHAR temp;
1219
1220 /* Note: This variable is only used on 30xLV systems.
1221 * CR38 has a different meaning on LVDS/CH7019 systems.
1222 * On 661 and later, these bits moved to CR35.
1223 *
1224 * On 301, 301B, only HiVision 1080i is supported.
1225 * On 30xLV, 301C, only YPbPr 1080i is supported.
1226 */
1227
1228 SiS_Pr->SiS_YPbPr = 0;
1229 if(HwInfo->jChipType >= SIS_661) return;
1230
1231 if(SiS_Pr->SiS_VBType) {
1232 if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) {
1233 SiS_Pr->SiS_YPbPr = YPbPrHiVision;
1234 }
1235 }
1236
1237 if(HwInfo->jChipType >= SIS_315H) {
1238 if(SiS_Pr->SiS_VBType & (VB_SIS301LV302LV | VB_SIS301C)) {
1239 temp = SiS_GetReg(SiS_Pr->SiS_P3d4,0x38);
1240 if(temp & 0x08) {
1241 switch((temp >> 4)) {
1242 case 0x00: SiS_Pr->SiS_YPbPr = YPbPr525i; break;
1243 case 0x01: SiS_Pr->SiS_YPbPr = YPbPr525p; break;
1244 case 0x02: SiS_Pr->SiS_YPbPr = YPbPr750p; break;
1245 case 0x03: SiS_Pr->SiS_YPbPr = YPbPrHiVision; break;
1246 }
1247 }
1248 }
1249 }
1250
1251}
1252
1253/*********************************************/
1254/* DETERMINE TVMode flag */
1255/*********************************************/
1256
1257void
1258SiS_SetTVMode(SiS_Private *SiS_Pr, USHORT ModeNo, USHORT ModeIdIndex, PSIS_HW_INFO HwInfo)
1259{
1260 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
1261 USHORT temp, temp1, resinfo = 0, romindex = 0;
1262 UCHAR OutputSelect = *SiS_Pr->pSiS_OutputSelect;
1263
1264 SiS_Pr->SiS_TVMode = 0;
1265
1266 if(!(SiS_Pr->SiS_VBInfo & SetCRT2ToTV)) return;
1267 if(SiS_Pr->UseCustomMode) return;
1268
1269 if(ModeNo > 0x13) {
1270 resinfo = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_RESINFO;
1271 }
1272
1273 if(HwInfo->jChipType < SIS_661) {
1274
1275 if(SiS_Pr->SiS_VBInfo & SetPALTV) SiS_Pr->SiS_TVMode |= TVSetPAL;
1276
1277 if(SiS_Pr->SiS_VBType & VB_SISVB) {
1278 temp = 0;
1279 if((HwInfo->jChipType == SIS_630) ||
1280 (HwInfo->jChipType == SIS_730)) {
1281 temp = 0x35;
1282 romindex = 0xfe;
1283 } else if(HwInfo->jChipType >= SIS_315H) {
1284 temp = 0x38;
1285 romindex = 0xf3;
1286 if(HwInfo->jChipType >= SIS_330) romindex = 0x11b;
1287 }
1288 if(temp) {
1289 if(romindex && SiS_Pr->SiS_UseROM && (!(SiS_Pr->SiS_ROMNew))) {
1290 OutputSelect = ROMAddr[romindex];
1291 if(!(OutputSelect & EnablePALMN)) {
1292 SiS_SetRegAND(SiS_Pr->SiS_P3d4,temp,0x3F);
1293 }
1294 }
1295 temp1 = SiS_GetReg(SiS_Pr->SiS_P3d4,temp);
1296 if(SiS_Pr->SiS_TVMode & TVSetPAL) {
1297 if(temp1 & EnablePALM) { /* 0x40 */
1298 SiS_Pr->SiS_TVMode |= TVSetPALM;
1299 SiS_Pr->SiS_TVMode &= ~TVSetPAL;
1300 } else if(temp1 & EnablePALN) { /* 0x80 */
1301 SiS_Pr->SiS_TVMode |= TVSetPALN;
1302 }
1303 } else {
1304 if(temp1 & EnableNTSCJ) { /* 0x40 */
1305 SiS_Pr->SiS_TVMode |= TVSetNTSCJ;
1306 }
1307 }
1308 }
1309 /* Translate HiVision/YPbPr to our new flags */
1310 if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) {
1311 if(SiS_Pr->SiS_YPbPr == YPbPr750p) SiS_Pr->SiS_TVMode |= TVSetYPbPr750p;
1312 else if(SiS_Pr->SiS_YPbPr == YPbPr525p) SiS_Pr->SiS_TVMode |= TVSetYPbPr525p;
1313 else if(SiS_Pr->SiS_YPbPr == YPbPrHiVision) SiS_Pr->SiS_TVMode |= TVSetHiVision;
1314 else SiS_Pr->SiS_TVMode |= TVSetYPbPr525i;
1315 if(SiS_Pr->SiS_TVMode & (TVSetYPbPr750p | TVSetYPbPr525p | TVSetYPbPr525i)) {
1316 SiS_Pr->SiS_VBInfo &= ~SetCRT2ToHiVision;
1317 SiS_Pr->SiS_VBInfo |= SetCRT2ToYPbPr525750;
1318 } else if(SiS_Pr->SiS_TVMode & TVSetHiVision) {
1319 SiS_Pr->SiS_TVMode |= TVSetPAL;
1320 }
1321 }
1322 } else if(SiS_Pr->SiS_IF_DEF_CH70xx != 0) {
1323 if(SiS_Pr->SiS_CHOverScan) {
1324 if(SiS_Pr->SiS_IF_DEF_CH70xx == 1) {
1325 temp = SiS_GetReg(SiS_Pr->SiS_P3d4,0x35);
1326 if((temp & TVOverScan) || (SiS_Pr->SiS_CHOverScan == 1)) {
1327 SiS_Pr->SiS_TVMode |= TVSetCHOverScan;
1328 }
1329 } else if(SiS_Pr->SiS_IF_DEF_CH70xx == 2) {
1330 temp = SiS_GetReg(SiS_Pr->SiS_P3d4,0x79);
1331 if((temp & 0x80) || (SiS_Pr->SiS_CHOverScan == 1)) {
1332 SiS_Pr->SiS_TVMode |= TVSetCHOverScan;
1333 }
1334 }
1335 if(SiS_Pr->SiS_CHSOverScan) {
1336 SiS_Pr->SiS_TVMode |= TVSetCHOverScan;
1337 }
1338 }
1339 if(SiS_Pr->SiS_IF_DEF_CH70xx == 2) {
1340 temp = SiS_GetReg(SiS_Pr->SiS_P3d4,0x38);
1341 if(SiS_Pr->SiS_TVMode & TVSetPAL) {
1342 if(temp & EnablePALM) SiS_Pr->SiS_TVMode |= TVSetPALM;
1343 else if(temp & EnablePALN) SiS_Pr->SiS_TVMode |= TVSetPALN;
1344 } else {
1345 if(temp & EnableNTSCJ) {
1346 SiS_Pr->SiS_TVMode |= TVSetNTSCJ;
1347 }
1348 }
1349 }
1350 }
1351
1352 } else { /* 661 and later */
1353
1354 temp1 = SiS_GetReg(SiS_Pr->SiS_P3d4,0x35);
1355 if(temp1 & 0x01) {
1356 SiS_Pr->SiS_TVMode |= TVSetPAL;
1357 if(temp1 & 0x08) {
1358 SiS_Pr->SiS_TVMode |= TVSetPALN;
1359 } else if(temp1 & 0x04) {
1360 if(SiS_Pr->SiS_VBType & VB_SISVB) {
1361 SiS_Pr->SiS_TVMode &= ~TVSetPAL;
1362 }
1363 SiS_Pr->SiS_TVMode |= TVSetPALM;
1364 }
1365 } else {
1366 if(temp1 & 0x02) {
1367 SiS_Pr->SiS_TVMode |= TVSetNTSCJ;
1368 }
1369 }
1370 if(SiS_Pr->SiS_IF_DEF_CH70xx == 2) {
1371 if(SiS_Pr->SiS_CHOverScan) {
1372 if((temp1 & 0x10) || (SiS_Pr->SiS_CHOverScan == 1)) {
1373 SiS_Pr->SiS_TVMode |= TVSetCHOverScan;
1374 }
1375 }
1376 }
1377 if(SiS_Pr->SiS_VBType & VB_SISVB) {
1378 if(SiS_Pr->SiS_VBInfo & SetCRT2ToYPbPr525750) {
1379 temp1 &= 0xe0;
1380 if(temp1 == 0x00) SiS_Pr->SiS_TVMode |= TVSetYPbPr525i;
1381 else if(temp1 == 0x20) SiS_Pr->SiS_TVMode |= TVSetYPbPr525p;
1382 else if(temp1 == 0x40) SiS_Pr->SiS_TVMode |= TVSetYPbPr750p;
1383 } else if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) {
1384 SiS_Pr->SiS_TVMode |= (TVSetHiVision | TVSetPAL);
1385 }
1386 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToYPbPr525750 | SetCRT2ToHiVision)) {
1387 if(resinfo == SIS_RI_800x480 || resinfo == SIS_RI_1024x576 || resinfo == SIS_RI_1280x720) {
1388 SiS_Pr->SiS_TVMode |= TVAspect169;
1389 } else {
1390 temp1 = SiS_GetReg(SiS_Pr->SiS_P3d4,0x39);
1391 if(temp1 & 0x02) {
1392 if(SiS_Pr->SiS_TVMode & (TVSetYPbPr750p | TVSetHiVision)) {
1393 SiS_Pr->SiS_TVMode |= TVAspect169;
1394 } else {
1395 SiS_Pr->SiS_TVMode |= TVAspect43LB;
1396 }
1397 } else {
1398 SiS_Pr->SiS_TVMode |= TVAspect43;
1399 }
1400 }
1401 }
1402 }
1403 }
1404
1405 if(SiS_Pr->SiS_VBInfo & SetCRT2ToSCART) SiS_Pr->SiS_TVMode |= TVSetPAL;
1406
1407 if(SiS_Pr->SiS_VBType & VB_SISVB) {
1408
1409 if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) {
1410 SiS_Pr->SiS_TVMode |= TVSetPAL;
1411 SiS_Pr->SiS_TVMode &= ~(TVSetPALM | TVSetPALN | TVSetNTSCJ);
1412 } else if(SiS_Pr->SiS_VBInfo & SetCRT2ToYPbPr525750) {
1413 if(SiS_Pr->SiS_TVMode & (TVSetYPbPr525i | TVSetYPbPr525p | TVSetYPbPr750p)) {
1414 SiS_Pr->SiS_TVMode &= ~(TVSetPAL | TVSetNTSCJ | TVSetPALM | TVSetPALN);
1415 }
1416 }
1417
1418 if(SiS_Pr->SiS_VBInfo & SetInSlaveMode) {
1419 if(!(SiS_Pr->SiS_VBInfo & SetNotSimuMode)) {
1420 SiS_Pr->SiS_TVMode |= TVSetTVSimuMode;
1421 }
1422 }
1423
1424 if(!(SiS_Pr->SiS_TVMode & TVSetPAL)) {
1425 /* BIOS sets TVNTSC1024 without checking 525p here. Wrong? */
1426 if(!(SiS_Pr->SiS_TVMode & (TVSetHiVision | TVSetYPbPr525p | TVSetYPbPr750p))) {
1427 if(resinfo == SIS_RI_1024x768) {
1428 SiS_Pr->SiS_TVMode |= TVSetNTSC1024;
1429 }
1430 }
1431 }
1432
1433 SiS_Pr->SiS_TVMode |= TVRPLLDIV2XO;
1434 if((SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) &&
1435 (SiS_Pr->SiS_VBInfo & SetInSlaveMode)) {
1436 SiS_Pr->SiS_TVMode &= ~TVRPLLDIV2XO;
1437 } else if(SiS_Pr->SiS_TVMode & (TVSetYPbPr525p | TVSetYPbPr750p)) {
1438 SiS_Pr->SiS_TVMode &= ~TVRPLLDIV2XO;
1439 } else if(!(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV)) {
1440 if(SiS_Pr->SiS_TVMode & TVSetTVSimuMode) {
1441 SiS_Pr->SiS_TVMode &= ~TVRPLLDIV2XO;
1442 }
1443 }
1444
1445 }
1446
1447 SiS_Pr->SiS_VBInfo &= ~SetPALTV;
1448
1449#ifdef TWDEBUG
1450 xf86DrvMsg(0, X_INFO, "(init301: TVMode %x, VBInfo %x)\n", SiS_Pr->SiS_TVMode, SiS_Pr->SiS_VBInfo);
1451#endif
1452}
1453
1454/*********************************************/
1455/* GET LCD INFO */
1456/*********************************************/
1457
1458static USHORT
1459SiS_GetBIOSLCDResInfo(SiS_Private *SiS_Pr)
1460{
1461 USHORT temp = SiS_Pr->SiS_LCDResInfo;
1462 /* Translate my LCDResInfo to BIOS value */
1463 if(temp == Panel_1280x768_2) temp = Panel_1280x768;
1464 if(temp == Panel_1280x800_2) temp = Panel_1280x800;
1465 return temp;
1466}
1467
1468static void
1469SiS_GetLCDInfoBIOS(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
1470{
1471#ifdef SIS315H
1472 UCHAR *ROMAddr;
1473 USHORT temp;
1474
1475#ifdef TWDEBUG
1476 xf86DrvMsg(0, X_INFO, "Paneldata driver: [%d %d] [H %d %d] [V %d %d] [C %d 0x%02x 0x%02x]\n",
1477 SiS_Pr->PanelHT, SiS_Pr->PanelVT,
1478 SiS_Pr->PanelHRS, SiS_Pr->PanelHRE,
1479 SiS_Pr->PanelVRS, SiS_Pr->PanelVRE,
1480 SiS_Pr->SiS_VBVCLKData[SiS_Pr->PanelVCLKIdx315].CLOCK,
1481 SiS_Pr->SiS_VBVCLKData[SiS_Pr->PanelVCLKIdx315].Part4_A,
1482 SiS_Pr->SiS_VBVCLKData[SiS_Pr->PanelVCLKIdx315].Part4_B);
1483#endif
1484
1485 if((ROMAddr = GetLCDStructPtr661(SiS_Pr, HwInfo))) {
1486 if((temp = SISGETROMW(6)) != SiS_Pr->PanelHT) {
1487 SiS_Pr->SiS_NeedRomModeData = TRUE;
1488 SiS_Pr->PanelHT = temp;
1489 }
1490 if((temp = SISGETROMW(8)) != SiS_Pr->PanelVT) {
1491 SiS_Pr->SiS_NeedRomModeData = TRUE;
1492 SiS_Pr->PanelVT = temp;
1493 }
1494 SiS_Pr->PanelHRS = SISGETROMW(10);
1495 SiS_Pr->PanelHRE = SISGETROMW(12);
1496 SiS_Pr->PanelVRS = SISGETROMW(14);
1497 SiS_Pr->PanelVRE = SISGETROMW(16);
1498 SiS_Pr->PanelVCLKIdx315 = VCLK_CUSTOM_315;
1499 SiS_Pr->SiS_VCLKData[VCLK_CUSTOM_315].CLOCK =
1500 SiS_Pr->SiS_VBVCLKData[VCLK_CUSTOM_315].CLOCK = (USHORT)((UCHAR)ROMAddr[18]);
1501 SiS_Pr->SiS_VCLKData[VCLK_CUSTOM_315].SR2B =
1502 SiS_Pr->SiS_VBVCLKData[VCLK_CUSTOM_315].Part4_A = ROMAddr[19];
1503 SiS_Pr->SiS_VCLKData[VCLK_CUSTOM_315].SR2C =
1504 SiS_Pr->SiS_VBVCLKData[VCLK_CUSTOM_315].Part4_B = ROMAddr[20];
1505
1506#ifdef TWDEBUG
1507 xf86DrvMsg(0, X_INFO, "Paneldata BIOS: [%d %d] [H %d %d] [V %d %d] [C %d 0x%02x 0x%02x]\n",
1508 SiS_Pr->PanelHT, SiS_Pr->PanelVT,
1509 SiS_Pr->PanelHRS, SiS_Pr->PanelHRE,
1510 SiS_Pr->PanelVRS, SiS_Pr->PanelVRE,
1511 SiS_Pr->SiS_VBVCLKData[SiS_Pr->PanelVCLKIdx315].CLOCK,
1512 SiS_Pr->SiS_VBVCLKData[SiS_Pr->PanelVCLKIdx315].Part4_A,
1513 SiS_Pr->SiS_VBVCLKData[SiS_Pr->PanelVCLKIdx315].Part4_B);
1514#endif
1515
1516 }
1517#endif
1518}
1519
1520static void
1521SiS_CheckScaling(SiS_Private *SiS_Pr, USHORT resinfo, const UCHAR *nonscalingmodes)
1522{
1523 int i = 0;
1524 while(nonscalingmodes[i] != 0xff) {
1525 if(nonscalingmodes[i++] == resinfo) {
1526 if((SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) ||
1527 (SiS_Pr->UsePanelScaler == -1)) {
1528 SiS_Pr->SiS_LCDInfo |= DontExpandLCD;
1529 }
1530 break;
1531 }
1532 }
1533}
1534
1535void
1536SiS_GetLCDResInfo(SiS_Private *SiS_Pr, USHORT ModeNo, USHORT ModeIdIndex,
1537 PSIS_HW_INFO HwInfo)
1538{
1539#ifdef SIS300
1540 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
1541 const unsigned char SiS300SeriesLCDRes[] =
1542 { 0, 1, 2, 3, 7, 4, 5, 8,
1543 0, 0, 10, 0, 0, 0, 0, 15 };
1544#endif
1545#ifdef SIS315H
1546 UCHAR *myptr = NULL;
1547#endif
1548 USHORT temp,modeflag,resinfo=0,modexres=0,modeyres=0;
1549 BOOLEAN panelcanscale = FALSE;
1550
1551 SiS_Pr->SiS_LCDResInfo = 0;
1552 SiS_Pr->SiS_LCDTypeInfo = 0;
1553 SiS_Pr->SiS_LCDInfo = 0;
1554 SiS_Pr->PanelHRS = 999; /* HSync start */
1555 SiS_Pr->PanelHRE = 999; /* HSync end */
1556 SiS_Pr->PanelVRS = 999; /* VSync start */
1557 SiS_Pr->PanelVRE = 999; /* VSync end */
1558 SiS_Pr->SiS_NeedRomModeData = FALSE;
1559
1560 if(!(SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA))) return;
1561
1562 if(ModeNo <= 0x13) {
1563 modeflag = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_ModeFlag;
1564 } else if(SiS_Pr->UseCustomMode) {
1565 modeflag = SiS_Pr->CModeFlag;
1566 } else {
1567 modeflag = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_ModeFlag;
1568 resinfo = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_RESINFO;
1569 modexres = SiS_Pr->SiS_ModeResInfo[resinfo].HTotal;
1570 modeyres = SiS_Pr->SiS_ModeResInfo[resinfo].VTotal;
1571 }
1572
1573 temp = SiS_GetReg(SiS_Pr->SiS_P3d4,0x36);
1574
1575 /* For broken BIOSes: Assume 1024x768 */
1576 if(temp == 0) temp = 0x02;
1577
1578 if((HwInfo->jChipType >= SIS_661) || (SiS_Pr->SiS_ROMNew)) {
1579 SiS_Pr->SiS_LCDTypeInfo = (SiS_GetReg(SiS_Pr->SiS_P3d4,0x39) & 0x7c) >> 2;
1580 } else if((HwInfo->jChipType < SIS_315H) || (HwInfo->jChipType >= SIS_661)) {
1581 SiS_Pr->SiS_LCDTypeInfo = temp >> 4;
1582 } else {
1583 SiS_Pr->SiS_LCDTypeInfo = (temp & 0x0F) - 1;
1584 }
1585 temp &= 0x0f;
1586#ifdef SIS300
1587 if(HwInfo->jChipType < SIS_315H) {
1588 /* Very old BIOSes only know 7 sizes (NetVista 2179, 1.01g) */
1589 if(SiS_Pr->SiS_VBType & VB_SIS301) {
1590 if(temp < 0x0f) temp &= 0x07;
1591 }
1592 /* Translate 300 series LCDRes to 315 series for unified usage */
1593 temp = SiS300SeriesLCDRes[temp];
1594 }
1595#endif
1596
1597 /* Translate to our internal types */
1598 if(HwInfo->jChipType == SIS_550) {
1599 if(temp == Panel310_640x480_2) temp = Panel_640x480_2;
1600 if(temp == Panel310_640x480_3) temp = Panel_640x480_3;
1601 }
1602
1603 if(SiS_Pr->SiS_VBType & VB_SISLVDS) { /* SiS LVDS */
1604 if(temp == Panel310_1280x768) {
1605 temp = Panel_1280x768_2;
1606 }
1607 if(SiS_Pr->SiS_ROMNew) {
1608 if(temp == Panel661_1280x800) {
1609 temp = Panel_1280x800_2;
1610 }
1611 }
1612 }
1613
1614 SiS_Pr->SiS_LCDResInfo = temp;
1615
1616 if(SiS_Pr->SiS_IF_DEF_LVDS == 1) {
1617 if(SiS_Pr->SiS_CustomT == CUT_BARCO1366) {
1618 SiS_Pr->SiS_LCDResInfo = Panel_Barco1366;
1619 } else if(SiS_Pr->SiS_CustomT == CUT_PANEL848) {
1620 SiS_Pr->SiS_LCDResInfo = Panel_848x480;
1621 }
1622 }
1623
1624 if(SiS_Pr->SiS_VBType & VB_SISVB) {
1625 if(SiS_Pr->SiS_LCDResInfo < SiS_Pr->SiS_PanelMin301)
1626 SiS_Pr->SiS_LCDResInfo = SiS_Pr->SiS_PanelMin301;
1627 } else {
1628 if(SiS_Pr->SiS_LCDResInfo < SiS_Pr->SiS_PanelMinLVDS)
1629 SiS_Pr->SiS_LCDResInfo = SiS_Pr->SiS_PanelMinLVDS;
1630 }
1631
1632 temp = SiS_GetReg(SiS_Pr->SiS_P3d4,0x37);
1633 SiS_Pr->SiS_LCDInfo = temp & ~0x000e;
1634 /* Need temp below! */
1635
1636 /* These can't scale no matter what */
1637 switch(SiS_Pr->SiS_LCDResInfo) {
1638 case Panel_1280x960:
1639 SiS_Pr->SiS_LCDInfo &= ~DontExpandLCD;
1640 }
1641
1642 panelcanscale = (SiS_Pr->SiS_LCDInfo & DontExpandLCD) ? TRUE : FALSE;
1643
1644 if(!SiS_Pr->UsePanelScaler) SiS_Pr->SiS_LCDInfo &= ~DontExpandLCD;
1645 else if(SiS_Pr->UsePanelScaler == 1) SiS_Pr->SiS_LCDInfo |= DontExpandLCD;
1646
1647 /* Dual link, Pass 1:1 BIOS default, etc. */
1648#ifdef SIS315H
1649 if(HwInfo->jChipType >= SIS_661) {
1650 if(SiS_Pr->SiS_LCDInfo & DontExpandLCD) {
1651 if(temp & 0x08) SiS_Pr->SiS_LCDInfo |= LCDPass11;
1652 }
1653 if(SiS_Pr->SiS_VBType & (VB_SIS302LV | VB_SIS302ELV)) {
1654 if(SiS_Pr->SiS_ROMNew) {
1655 if(temp & 0x02) SiS_Pr->SiS_LCDInfo |= LCDDualLink;
1656 } else if((myptr = GetLCDStructPtr661(SiS_Pr, HwInfo))) {
1657 if(myptr[2] & 0x01) SiS_Pr->SiS_LCDInfo |= LCDDualLink;
1658 }
1659 }
1660 } else if(HwInfo->jChipType >= SIS_315H) {
1661 if(SiS_Pr->SiS_LCDInfo & DontExpandLCD) {
1662 if(SiS_GetReg(SiS_Pr->SiS_P3d4,0x39) & 0x01) SiS_Pr->SiS_LCDInfo |= LCDPass11;
1663 }
1664 if((SiS_Pr->SiS_ROMNew) && (!(SiS_Pr->PanelSelfDetected))) {
1665 SiS_Pr->SiS_LCDInfo &= ~(LCDRGB18Bit);
1666 temp = SiS_GetReg(SiS_Pr->SiS_P3d4,0x35);
1667 if(temp & 0x01) SiS_Pr->SiS_LCDInfo |= LCDRGB18Bit;
1668 if(SiS_Pr->SiS_VBType & (VB_SIS302LV | VB_SIS302ELV)) {
1669 if(temp & 0x02) SiS_Pr->SiS_LCDInfo |= LCDDualLink;
1670 }
1671 } else if(!(SiS_Pr->SiS_ROMNew)) {
1672 if(SiS_Pr->SiS_VBType & (VB_SIS302LV | VB_SIS302ELV)) {
1673 if((SiS_Pr->SiS_CustomT == CUT_CLEVO1024) &&
1674 (SiS_Pr->SiS_LCDResInfo == Panel_1024x768)) {
1675 SiS_Pr->SiS_LCDInfo |= LCDDualLink;
1676 }
1677 if((SiS_Pr->SiS_LCDResInfo == Panel_1280x1024) ||
1678 (SiS_Pr->SiS_LCDResInfo == Panel_1400x1050) ||
1679 (SiS_Pr->SiS_LCDResInfo == Panel_1600x1200) ||
1680 (SiS_Pr->SiS_LCDResInfo == Panel_1680x1050)) {
1681 SiS_Pr->SiS_LCDInfo |= LCDDualLink;
1682 }
1683 }
1684 }
1685 }
1686#endif
1687
1688 /* Pass 1:1 */
1689 if((SiS_Pr->SiS_IF_DEF_LVDS == 1) || (SiS_Pr->SiS_VBType & VB_NoLCD)) {
1690 /* Always center screen on LVDS (if scaling is disabled) */
1691 SiS_Pr->SiS_LCDInfo &= ~LCDPass11;
1692 } else if(SiS_Pr->SiS_VBType & VB_SISVB) {
1693 if(SiS_Pr->SiS_VBType & VB_SISLVDS) {
1694 /* Always center screen on SiS LVDS (if scaling is disabled) */
1695 SiS_Pr->SiS_LCDInfo &= ~LCDPass11;
1696 } else {
1697 /* By default, pass 1:1 on SiS TMDS (if scaling is supported) */
1698 if(panelcanscale) SiS_Pr->SiS_LCDInfo |= LCDPass11;
1699 if(SiS_Pr->CenterScreen == 1) SiS_Pr->SiS_LCDInfo &= ~LCDPass11;
1700 }
1701 }
1702
1703 SiS_Pr->PanelVCLKIdx300 = VCLK65_300;
1704 SiS_Pr->PanelVCLKIdx315 = VCLK108_2_315;
1705
1706 switch(SiS_Pr->SiS_LCDResInfo) {
1707 case Panel_320x480: SiS_Pr->PanelXRes = 320; SiS_Pr->PanelYRes = 480;
1708 SiS_Pr->PanelHT = 400; SiS_Pr->PanelVT = 525;
1709 SiS_Pr->PanelVCLKIdx300 = VCLK28;
1710 SiS_Pr->PanelVCLKIdx315 = VCLK28;
1711 break;
1712 case Panel_640x480_2:
1713 case Panel_640x480_3: SiS_Pr->PanelXRes = 640; SiS_Pr->PanelYRes = 480;
1714 SiS_Pr->PanelVRS = 24; SiS_Pr->PanelVRE = 3;
1715 SiS_Pr->PanelVCLKIdx300 = VCLK28;
1716 SiS_Pr->PanelVCLKIdx315 = VCLK28;
1717 break;
1718 case Panel_640x480: SiS_Pr->PanelXRes = 640; SiS_Pr->PanelYRes = 480;
1719 SiS_Pr->PanelVRE = 3;
1720 SiS_Pr->PanelVCLKIdx300 = VCLK28;
1721 SiS_Pr->PanelVCLKIdx315 = VCLK28;
1722 break;
1723 case Panel_800x600: SiS_Pr->PanelXRes = 800; SiS_Pr->PanelYRes = 600;
1724 SiS_Pr->PanelHT = 1056; SiS_Pr->PanelVT = 628;
1725 SiS_Pr->PanelHRS = 40; SiS_Pr->PanelHRE = 128;
1726 SiS_Pr->PanelVRS = 1; SiS_Pr->PanelVRE = 4;
1727 SiS_Pr->PanelVCLKIdx300 = VCLK40;
1728 SiS_Pr->PanelVCLKIdx315 = VCLK40;
1729 break;
1730 case Panel_1024x600: SiS_Pr->PanelXRes = 1024; SiS_Pr->PanelYRes = 600;
1731 SiS_Pr->PanelHT = 1344; SiS_Pr->PanelVT = 800;
1732 SiS_Pr->PanelHRS = 24; SiS_Pr->PanelHRE = 136;
1733 SiS_Pr->PanelVRS = 2 /* 88 */ ; SiS_Pr->PanelVRE = 6;
1734 SiS_Pr->PanelVCLKIdx300 = VCLK65_300;
1735 SiS_Pr->PanelVCLKIdx315 = VCLK65_315;
1736 break;
1737 case Panel_1024x768: SiS_Pr->PanelXRes = 1024; SiS_Pr->PanelYRes = 768;
1738 SiS_Pr->PanelHT = 1344; SiS_Pr->PanelVT = 806;
1739 SiS_Pr->PanelHRS = 24; SiS_Pr->PanelHRE = 136;
1740 SiS_Pr->PanelVRS = 3; SiS_Pr->PanelVRE = 6;
1741 if(HwInfo->jChipType < SIS_315H) {
1742 SiS_Pr->PanelHRS = 23;
1743 SiS_Pr->PanelVRE = 5;
1744 }
1745 SiS_Pr->PanelVCLKIdx300 = VCLK65_300;
1746 SiS_Pr->PanelVCLKIdx315 = VCLK65_315;
1747 SiS_GetLCDInfoBIOS(SiS_Pr, HwInfo);
1748 break;
1749 case Panel_1152x768: SiS_Pr->PanelXRes = 1152; SiS_Pr->PanelYRes = 768;
1750 SiS_Pr->PanelHT = 1344; SiS_Pr->PanelVT = 806;
1751 SiS_Pr->PanelHRS = 24;
1752 SiS_Pr->PanelVRS = 3; SiS_Pr->PanelVRE = 6;
1753 if(HwInfo->jChipType < SIS_315H) {
1754 SiS_Pr->PanelHRS = 23;
1755 SiS_Pr->PanelVRE = 5;
1756 }
1757 SiS_Pr->PanelVCLKIdx300 = VCLK65_300;
1758 SiS_Pr->PanelVCLKIdx315 = VCLK65_315;
1759 break;
1760 case Panel_1152x864: SiS_Pr->PanelXRes = 1152; SiS_Pr->PanelYRes = 864;
1761 break;
1762 case Panel_1280x720: SiS_Pr->PanelXRes = 1280; SiS_Pr->PanelYRes = 720;
1763 SiS_Pr->PanelHT = 1650; SiS_Pr->PanelVT = 750;
1764 SiS_Pr->PanelHRS = 110; SiS_Pr->PanelHRE = 40;
1765 SiS_Pr->PanelVRS = 5; SiS_Pr->PanelVRE = 5;
1766 SiS_Pr->PanelVCLKIdx315 = VCLK_1280x720;
1767 /* Data above for TMDS (projector); get from BIOS for LVDS */
1768 SiS_GetLCDInfoBIOS(SiS_Pr, HwInfo);
1769 break;
1770 case Panel_1280x768: SiS_Pr->PanelXRes = 1280; SiS_Pr->PanelYRes = 768;
1771 if(SiS_Pr->SiS_IF_DEF_LVDS == 1) {
1772 SiS_Pr->PanelHT = 1408; SiS_Pr->PanelVT = 806;
1773 SiS_Pr->PanelVCLKIdx300 = VCLK81_300; /* ? */
1774 SiS_Pr->PanelVCLKIdx315 = VCLK81_315; /* ? */
1775 } else {
1776 SiS_Pr->PanelHT = 1688; SiS_Pr->PanelVT = 802;
1777 SiS_Pr->PanelHRS = 48; SiS_Pr->PanelHRS = 112;
1778 SiS_Pr->PanelVRS = 3; SiS_Pr->PanelVRE = 6;
1779 SiS_Pr->PanelVCLKIdx300 = VCLK81_300;
1780 SiS_Pr->PanelVCLKIdx315 = VCLK81_315;
1781 }
1782 break;
1783 case Panel_1280x768_2: SiS_Pr->PanelXRes = 1280; SiS_Pr->PanelYRes = 768;
1784 SiS_Pr->PanelHT = 1660; SiS_Pr->PanelVT = 806;
1785 SiS_Pr->PanelHRS = 48; SiS_Pr->PanelHRE = 112;
1786 SiS_Pr->PanelVRS = 3; SiS_Pr->PanelVRE = 6;
1787 SiS_Pr->PanelVCLKIdx315 = VCLK_1280x768_2;
1788 SiS_GetLCDInfoBIOS(SiS_Pr, HwInfo);
1789 break;
1790 case Panel_1280x800: SiS_Pr->PanelXRes = 1280; SiS_Pr->PanelYRes = 800;
1791 SiS_Pr->PanelHT = 1408; SiS_Pr->PanelVT = 816;
1792 SiS_Pr->PanelHRS = 21; SiS_Pr->PanelHRE = 24;
1793 SiS_Pr->PanelVRS = 4; SiS_Pr->PanelVRE = 3;
1794 SiS_Pr->PanelVCLKIdx315 = VCLK_1280x800_315;
1795 SiS_GetLCDInfoBIOS(SiS_Pr, HwInfo);
1796 break;
1797 case Panel_1280x800_2: SiS_Pr->PanelXRes = 1280; SiS_Pr->PanelYRes = 800;
1798 SiS_Pr->PanelHT = 1552; SiS_Pr->PanelVT = 812;
1799 SiS_Pr->PanelHRS = 48; SiS_Pr->PanelHRE = 112;
1800 SiS_Pr->PanelVRS = 4; SiS_Pr->PanelVRE = 3;
1801 SiS_Pr->PanelVCLKIdx315 = VCLK_1280x800_315_2;
1802 SiS_GetLCDInfoBIOS(SiS_Pr, HwInfo);
1803 break;
1804 case Panel_1280x960: SiS_Pr->PanelXRes = 1280; SiS_Pr->PanelYRes = 960;
1805 SiS_Pr->PanelHT = 1800; SiS_Pr->PanelVT = 1000;
1806 SiS_Pr->PanelVCLKIdx300 = VCLK108_3_300;
1807 SiS_Pr->PanelVCLKIdx315 = VCLK108_3_315;
1808 if(resinfo == SIS_RI_1280x1024) {
1809 SiS_Pr->PanelVCLKIdx300 = VCLK100_300;
1810 SiS_Pr->PanelVCLKIdx315 = VCLK100_315;
1811 }
1812 break;
1813 case Panel_1280x1024: SiS_Pr->PanelXRes = 1280; SiS_Pr->PanelYRes = 1024;
1814 SiS_Pr->PanelHT = 1688; SiS_Pr->PanelVT = 1066;
1815 SiS_Pr->PanelHRS = 48; SiS_Pr->PanelHRE = 112;
1816 SiS_Pr->PanelVRS = 1; SiS_Pr->PanelVRE = 3;
1817 SiS_Pr->PanelVCLKIdx300 = VCLK108_3_300;
1818 SiS_Pr->PanelVCLKIdx315 = VCLK108_2_315;
1819 SiS_GetLCDInfoBIOS(SiS_Pr, HwInfo);
1820 break;
1821 case Panel_1400x1050: SiS_Pr->PanelXRes = 1400; SiS_Pr->PanelYRes = 1050;
1822 SiS_Pr->PanelHT = 1688; SiS_Pr->PanelVT = 1066;
1823 SiS_Pr->PanelHRS = 48; SiS_Pr->PanelHRE = 112; /* HRE OK for LVDS, not for LCDA */
1824 SiS_Pr->PanelVRS = 1; SiS_Pr->PanelVRE = 3;
1825 SiS_Pr->PanelVCLKIdx315 = VCLK108_2_315;
1826 SiS_GetLCDInfoBIOS(SiS_Pr, HwInfo);
1827 break;
1828 case Panel_1600x1200: SiS_Pr->PanelXRes = 1600; SiS_Pr->PanelYRes = 1200;
1829 SiS_Pr->PanelHT = 2160; SiS_Pr->PanelVT = 1250;
1830 SiS_Pr->PanelHRS = 64; SiS_Pr->PanelHRE = 192;
1831 SiS_Pr->PanelVRS = 1; SiS_Pr->PanelVRE = 3;
1832 SiS_Pr->PanelVCLKIdx315 = VCLK162_315;
1833 SiS_GetLCDInfoBIOS(SiS_Pr, HwInfo);
1834 break;
1835 case Panel_1680x1050: SiS_Pr->PanelXRes = 1680; SiS_Pr->PanelYRes = 1050;
1836 SiS_Pr->PanelHT = 1900; SiS_Pr->PanelVT = 1066;
1837 SiS_Pr->PanelHRS = 26; SiS_Pr->PanelHRE = 76;
1838 SiS_Pr->PanelVRS = 3; SiS_Pr->PanelVRE = 6;
1839 SiS_Pr->PanelVCLKIdx315 = VCLK121_315;
1840 SiS_GetLCDInfoBIOS(SiS_Pr, HwInfo);
1841 break;
1842 case Panel_Barco1366: SiS_Pr->PanelXRes = 1360; SiS_Pr->PanelYRes = 1024;
1843 SiS_Pr->PanelHT = 1688; SiS_Pr->PanelVT = 1066;
1844 break;
1845 case Panel_848x480: SiS_Pr->PanelXRes = 848; SiS_Pr->PanelYRes = 480;
1846 SiS_Pr->PanelHT = 1088; SiS_Pr->PanelVT = 525;
1847 break;
1848 case Panel_Custom: SiS_Pr->PanelXRes = SiS_Pr->CP_MaxX;
1849 SiS_Pr->PanelYRes = SiS_Pr->CP_MaxY;
1850 SiS_Pr->PanelHT = SiS_Pr->CHTotal;
1851 SiS_Pr->PanelVT = SiS_Pr->CVTotal;
1852 if(SiS_Pr->CP_PreferredIndex != -1) {
1853 SiS_Pr->PanelXRes = SiS_Pr->CP_HDisplay[SiS_Pr->CP_PreferredIndex];
1854 SiS_Pr->PanelYRes = SiS_Pr->CP_VDisplay[SiS_Pr->CP_PreferredIndex];
1855 SiS_Pr->PanelHT = SiS_Pr->CP_HTotal[SiS_Pr->CP_PreferredIndex];
1856 SiS_Pr->PanelVT = SiS_Pr->CP_VTotal[SiS_Pr->CP_PreferredIndex];
1857 SiS_Pr->PanelHRS = SiS_Pr->CP_HSyncStart[SiS_Pr->CP_PreferredIndex];
1858 SiS_Pr->PanelHRE = SiS_Pr->CP_HSyncEnd[SiS_Pr->CP_PreferredIndex];
1859 SiS_Pr->PanelVRS = SiS_Pr->CP_VSyncStart[SiS_Pr->CP_PreferredIndex];
1860 SiS_Pr->PanelVRE = SiS_Pr->CP_VSyncEnd[SiS_Pr->CP_PreferredIndex];
1861 SiS_Pr->PanelHRS -= SiS_Pr->PanelXRes;
1862 SiS_Pr->PanelHRE -= SiS_Pr->PanelHRS;
1863 SiS_Pr->PanelVRS -= SiS_Pr->PanelYRes;
1864 SiS_Pr->PanelVRE -= SiS_Pr->PanelVRS;
1865 if(SiS_Pr->CP_PrefClock) {
1866 int idx;
1867 SiS_Pr->PanelVCLKIdx315 = VCLK_CUSTOM_315;
1868 SiS_Pr->PanelVCLKIdx300 = VCLK_CUSTOM_300;
1869 if(HwInfo->jChipType < SIS_315H) idx = VCLK_CUSTOM_300;
1870 else idx = VCLK_CUSTOM_315;
1871 SiS_Pr->SiS_VCLKData[idx].CLOCK =
1872 SiS_Pr->SiS_VBVCLKData[idx].CLOCK = SiS_Pr->CP_PrefClock;
1873 SiS_Pr->SiS_VCLKData[idx].SR2B =
1874 SiS_Pr->SiS_VBVCLKData[idx].Part4_A = SiS_Pr->CP_PrefSR2B;
1875 SiS_Pr->SiS_VCLKData[idx].SR2C =
1876 SiS_Pr->SiS_VBVCLKData[idx].Part4_B = SiS_Pr->CP_PrefSR2C;
1877 }
1878 }
1879 break;
1880 default: SiS_Pr->PanelXRes = 1024; SiS_Pr->PanelYRes = 768;
1881 SiS_Pr->PanelHT = 1344; SiS_Pr->PanelVT = 806;
1882 break;
1883 }
1884
1885 /* Special cases */
1886 if( (SiS_Pr->SiS_IF_DEF_FSTN) ||
1887 (SiS_Pr->SiS_IF_DEF_DSTN) ||
1888 (SiS_Pr->SiS_CustomT == CUT_BARCO1366) ||
1889 (SiS_Pr->SiS_CustomT == CUT_BARCO1024) ||
1890 (SiS_Pr->SiS_CustomT == CUT_PANEL848) ) {
1891 SiS_Pr->PanelHRS = 999;
1892 SiS_Pr->PanelHRE = 999;
1893 }
1894
1895 if( (SiS_Pr->SiS_CustomT == CUT_BARCO1366) ||
1896 (SiS_Pr->SiS_CustomT == CUT_BARCO1024) ||
1897 (SiS_Pr->SiS_CustomT == CUT_PANEL848) ) {
1898 SiS_Pr->PanelVRS = 999;
1899 SiS_Pr->PanelVRE = 999;
1900 }
1901
1902 /* DontExpand overrule */
1903 if((SiS_Pr->SiS_VBType & VB_SISVB) && (!(SiS_Pr->SiS_VBType & VB_NoLCD))) {
1904
1905 if((SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) && (modeflag & NoSupportLCDScale)) {
1906 /* No scaling for this mode on any panel (LCD=CRT2)*/
1907 SiS_Pr->SiS_LCDInfo |= DontExpandLCD;
1908 }
1909
1910 switch(SiS_Pr->SiS_LCDResInfo) {
1911
1912 case Panel_Custom:
1913 case Panel_1152x864:
1914 case Panel_1280x768: /* TMDS only */
1915 SiS_Pr->SiS_LCDInfo |= DontExpandLCD;
1916 break;
1917
1918 case Panel_800x600: {
1919 static const UCHAR nonscalingmodes[] = {
1920 SIS_RI_720x480, SIS_RI_720x576, SIS_RI_768x576, SIS_RI_800x480, 0xff
1921 };
1922 SiS_CheckScaling(SiS_Pr, resinfo, nonscalingmodes);
1923 break;
1924 }
1925 case Panel_1024x768: {
1926 static const UCHAR nonscalingmodes[] = {
1927 SIS_RI_720x480, SIS_RI_720x576, SIS_RI_768x576, SIS_RI_800x480, SIS_RI_848x480,
1928 SIS_RI_856x480, SIS_RI_960x540, SIS_RI_960x600, SIS_RI_1024x576,SIS_RI_1024x600,
1929 0xff
1930 };
1931 SiS_CheckScaling(SiS_Pr, resinfo, nonscalingmodes);
1932 break;
1933 }
1934 case Panel_1280x720: {
1935 static const UCHAR nonscalingmodes[] = {
1936 SIS_RI_720x480, SIS_RI_720x576, SIS_RI_768x576, SIS_RI_800x480, SIS_RI_848x480,
1937 SIS_RI_856x480, SIS_RI_960x540, SIS_RI_960x600, SIS_RI_1024x576,SIS_RI_1024x600,
1938 0xff
1939 };
1940 SiS_CheckScaling(SiS_Pr, resinfo, nonscalingmodes);
1941 if(SiS_Pr->PanelHT == 1650) {
1942 SiS_Pr->SiS_LCDInfo |= DontExpandLCD;
1943 }
1944 break;
1945 }
1946 case Panel_1280x768_2: { /* LVDS only */
1947 static const UCHAR nonscalingmodes[] = {
1948 SIS_RI_720x480, SIS_RI_720x576, SIS_RI_768x576, SIS_RI_800x480, SIS_RI_848x480,
1949 SIS_RI_856x480, SIS_RI_960x540, SIS_RI_960x600, SIS_RI_1024x576,SIS_RI_1024x600,
1950 SIS_RI_1152x768,0xff
1951 };
1952 SiS_CheckScaling(SiS_Pr, resinfo, nonscalingmodes);
1953 switch(resinfo) {
1954 case SIS_RI_1280x720: if(SiS_Pr->UsePanelScaler == -1) {
1955 SiS_Pr->SiS_LCDInfo |= DontExpandLCD;
1956 }
1957 break;
1958 }
1959 break;
1960 }
1961 case Panel_1280x800: { /* SiS TMDS special (Averatec 6200 series) */
1962 static const UCHAR nonscalingmodes[] = {
1963 SIS_RI_720x480, SIS_RI_720x576, SIS_RI_768x576, SIS_RI_800x480, SIS_RI_848x480,
1964 SIS_RI_856x480, SIS_RI_960x540, SIS_RI_960x600, SIS_RI_1024x576,SIS_RI_1024x600,
1965 SIS_RI_1152x768,SIS_RI_1280x720,SIS_RI_1280x768,0xff
1966 };
1967 SiS_CheckScaling(SiS_Pr, resinfo, nonscalingmodes);
1968 break;
1969 }
1970 case Panel_1280x800_2: { /* SiS LVDS */
1971 static const UCHAR nonscalingmodes[] = {
1972 SIS_RI_720x480, SIS_RI_720x576, SIS_RI_768x576, SIS_RI_800x480, SIS_RI_848x480,
1973 SIS_RI_856x480, SIS_RI_960x540, SIS_RI_960x600, SIS_RI_1024x576,SIS_RI_1024x600,
1974 SIS_RI_1152x768,0xff
1975 };
1976 SiS_CheckScaling(SiS_Pr, resinfo, nonscalingmodes);
1977 switch(resinfo) {
1978 case SIS_RI_1280x720:
1979 case SIS_RI_1280x768: if(SiS_Pr->UsePanelScaler == -1) {
1980 SiS_Pr->SiS_LCDInfo |= DontExpandLCD;
1981 }
1982 break;
1983 }
1984 break;
1985 }
1986 case Panel_1280x960: {
1987 static const UCHAR nonscalingmodes[] = {
1988 SIS_RI_720x480, SIS_RI_720x576, SIS_RI_768x576, SIS_RI_800x480, SIS_RI_848x480,
1989 SIS_RI_856x480, SIS_RI_960x540, SIS_RI_960x600, SIS_RI_1024x576,SIS_RI_1024x600,
1990 SIS_RI_1152x768,SIS_RI_1152x864,SIS_RI_1280x720,SIS_RI_1280x768,SIS_RI_1280x800,
1991 0xff
1992 };
1993 SiS_CheckScaling(SiS_Pr, resinfo, nonscalingmodes);
1994 break;
1995 }
1996 case Panel_1280x1024: {
1997 static const UCHAR nonscalingmodes[] = {
1998 SIS_RI_720x480, SIS_RI_720x576, SIS_RI_768x576, SIS_RI_800x480, SIS_RI_848x480,
1999 SIS_RI_856x480, SIS_RI_960x540, SIS_RI_960x600, SIS_RI_1024x576,SIS_RI_1024x600,
2000 SIS_RI_1152x768,SIS_RI_1152x864,SIS_RI_1280x720,SIS_RI_1280x768,SIS_RI_1280x800,
2001 SIS_RI_1280x960,0xff
2002 };
2003 SiS_CheckScaling(SiS_Pr, resinfo, nonscalingmodes);
2004 break;
2005 }
2006 case Panel_1400x1050: {
2007 static const UCHAR nonscalingmodes[] = {
2008 SIS_RI_720x480, SIS_RI_720x576, SIS_RI_768x576, SIS_RI_800x480, SIS_RI_848x480,
2009 SIS_RI_856x480, SIS_RI_960x540, SIS_RI_960x600, SIS_RI_1024x576,SIS_RI_1024x600,
2010 SIS_RI_1152x768,SIS_RI_1152x864,SIS_RI_1280x768,SIS_RI_1280x800,SIS_RI_1280x960,
2011 0xff
2012 };
2013 SiS_CheckScaling(SiS_Pr, resinfo, nonscalingmodes);
2014 switch(resinfo) {
2015 case SIS_RI_1280x720: if(SiS_Pr->UsePanelScaler == -1) {
2016 SiS_Pr->SiS_LCDInfo |= DontExpandLCD;
2017 }
2018 break;
2019 case SIS_RI_1280x1024: SiS_Pr->SiS_LCDInfo |= DontExpandLCD;
2020 break;
2021 }
2022 break;
2023 }
2024 case Panel_1600x1200: {
2025 static const UCHAR nonscalingmodes[] = {
2026 SIS_RI_720x480, SIS_RI_720x576, SIS_RI_768x576, SIS_RI_800x480, SIS_RI_848x480,
2027 SIS_RI_856x480, SIS_RI_960x540, SIS_RI_960x600, SIS_RI_1024x576,SIS_RI_1024x600,
2028 SIS_RI_1152x768,SIS_RI_1152x864,SIS_RI_1280x720,SIS_RI_1280x768,SIS_RI_1280x800,
2029 SIS_RI_1280x960,SIS_RI_1360x768,SIS_RI_1360x1024,0xff
2030 };
2031 SiS_CheckScaling(SiS_Pr, resinfo, nonscalingmodes);
2032 break;
2033 }
2034 case Panel_1680x1050: {
2035 static const UCHAR nonscalingmodes[] = {
2036 SIS_RI_720x480, SIS_RI_720x576, SIS_RI_768x576, SIS_RI_800x480, SIS_RI_848x480,
2037 SIS_RI_856x480, SIS_RI_960x540, SIS_RI_960x600, SIS_RI_1024x576,SIS_RI_1024x600,
2038 SIS_RI_1152x768,SIS_RI_1152x864,SIS_RI_1280x960,SIS_RI_1360x768,SIS_RI_1360x1024,
2039 0xff
2040 };
2041 SiS_CheckScaling(SiS_Pr, resinfo, nonscalingmodes);
2042 break;
2043 }
2044 }
2045 }
2046
2047 if(SiS_Pr->SiS_IF_DEF_LVDS == 1) {
2048 if(SiS_Pr->SiS_CustomT == CUT_PANEL848) {
2049 SiS_Pr->SiS_LCDInfo = 0x80 | 0x40 | 0x20; /* neg h/v sync, RGB24(D0 = 0) */
2050 }
2051 }
2052
2053#ifdef SIS300
2054 if(HwInfo->jChipType < SIS_315H) {
2055 if(SiS_Pr->SiS_IF_DEF_LVDS == 1) {
2056 if(SiS_Pr->SiS_UseROM) {
2057 if((ROMAddr[0x233] == 0x12) && (ROMAddr[0x234] == 0x34)) {
2058 if(!(ROMAddr[0x235] & 0x02)) {
2059 SiS_Pr->SiS_LCDInfo &= (~DontExpandLCD);
2060 }
2061 }
2062 }
2063 } else if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) {
2064 if((SiS_Pr->SiS_SetFlag & SetDOSMode) && ((ModeNo == 0x03) || (ModeNo == 0x10))) {
2065 SiS_Pr->SiS_LCDInfo &= (~DontExpandLCD);
2066 }
2067 }
2068 }
2069#endif
2070
2071 /* Special cases */
2072
2073 if(modexres == SiS_Pr->PanelXRes && modeyres == SiS_Pr->PanelYRes) {
2074 SiS_Pr->SiS_LCDInfo &= ~LCDPass11;
2075 }
2076
2077 if(SiS_Pr->SiS_IF_DEF_TRUMPION) {
2078 SiS_Pr->SiS_LCDInfo |= (DontExpandLCD | LCDPass11);
2079 }
2080
2081 switch(SiS_Pr->SiS_LCDResInfo) {
2082 case Panel_640x480:
2083 SiS_Pr->SiS_LCDInfo |= LCDPass11;
2084 break;
2085 case Panel_1280x800:
2086 /* Don't pass 1:1 by default (TMDS special) */
2087 if(SiS_Pr->CenterScreen == -1) SiS_Pr->SiS_LCDInfo &= ~LCDPass11;
2088 break;
2089 case Panel_1280x960:
2090 SiS_Pr->SiS_LCDInfo &= ~LCDPass11;
2091 break;
2092 case Panel_Custom:
2093 if((!SiS_Pr->CP_PrefClock) ||
2094 (modexres > SiS_Pr->PanelXRes) || (modeyres > SiS_Pr->PanelYRes)) {
2095 SiS_Pr->SiS_LCDInfo |= LCDPass11;
2096 }
2097 break;
2098 }
2099
2100 if(SiS_Pr->UseCustomMode) {
2101 SiS_Pr->SiS_LCDInfo |= (DontExpandLCD | LCDPass11);
2102 }
2103
2104 /* (In)validate LCDPass11 flag */
2105 if(!(SiS_Pr->SiS_LCDInfo & DontExpandLCD)) {
2106 SiS_Pr->SiS_LCDInfo &= ~LCDPass11;
2107 }
2108
2109 /* LVDS DDA */
2110 if(!((HwInfo->jChipType < SIS_315H) && (SiS_Pr->SiS_SetFlag & SetDOSMode))) {
2111
2112 if((SiS_Pr->SiS_IF_DEF_LVDS == 1) || (SiS_Pr->SiS_VBType & VB_NoLCD)) {
2113 if(SiS_Pr->SiS_IF_DEF_TRUMPION == 0) {
2114 if(ModeNo == 0x12) {
2115 if(SiS_Pr->SiS_LCDInfo & LCDPass11) {
2116 SiS_Pr->SiS_SetFlag |= EnableLVDSDDA;
2117 }
2118 } else if(ModeNo > 0x13) {
2119 if(SiS_Pr->SiS_LCDResInfo == Panel_1024x600) {
2120 if(!(SiS_Pr->SiS_LCDInfo & DontExpandLCD)) {
2121 if((resinfo == SIS_RI_800x600) || (resinfo == SIS_RI_400x300)) {
2122 SiS_Pr->SiS_SetFlag |= EnableLVDSDDA;
2123 }
2124 }
2125 }
2126 }
2127 }
2128 }
2129
2130 if(modeflag & HalfDCLK) {
2131 if(SiS_Pr->SiS_IF_DEF_TRUMPION == 1) {
2132 SiS_Pr->SiS_SetFlag |= EnableLVDSDDA;
2133 } else if(SiS_Pr->SiS_LCDInfo & DontExpandLCD) {
2134 SiS_Pr->SiS_SetFlag |= EnableLVDSDDA;
2135 } else if(SiS_Pr->SiS_LCDResInfo == Panel_640x480) {
2136 SiS_Pr->SiS_SetFlag |= EnableLVDSDDA;
2137 } else if(ModeNo > 0x13) {
2138 if(SiS_Pr->SiS_LCDResInfo == Panel_1024x768) {
2139 if(resinfo == SIS_RI_512x384) SiS_Pr->SiS_SetFlag |= EnableLVDSDDA;
2140 } else if(SiS_Pr->SiS_LCDResInfo == Panel_800x600) {
2141 if(resinfo == SIS_RI_400x300) SiS_Pr->SiS_SetFlag |= EnableLVDSDDA;
2142 }
2143 }
2144 }
2145
2146 }
2147
2148 /* VESA timing */
2149 if(SiS_Pr->SiS_VBInfo & SetInSlaveMode) {
2150 if(SiS_Pr->SiS_VBInfo & SetNotSimuMode) {
2151 SiS_Pr->SiS_SetFlag |= LCDVESATiming;
2152 }
2153 } else {
2154 SiS_Pr->SiS_SetFlag |= LCDVESATiming;
2155 }
2156
2157#ifdef LINUX_KERNEL
2158#ifdef TWDEBUG
2159 printk(KERN_DEBUG "sisfb: (LCDInfo=0x%04x LCDResInfo=0x%02x LCDTypeInfo=0x%02x)\n",
2160 SiS_Pr->SiS_LCDInfo, SiS_Pr->SiS_LCDResInfo, SiS_Pr->SiS_LCDTypeInfo);
2161#endif
2162#endif
2163#ifdef LINUX_XF86
2164 xf86DrvMsgVerb(0, X_PROBED, 4,
2165 "(init301: LCDInfo=0x%04x LCDResInfo=0x%02x LCDTypeInfo=0x%02x SetFlag=0x%04x)\n",
2166 SiS_Pr->SiS_LCDInfo, SiS_Pr->SiS_LCDResInfo, SiS_Pr->SiS_LCDTypeInfo, SiS_Pr->SiS_SetFlag);
2167#endif
2168}
2169
2170/*********************************************/
2171/* GET VCLK */
2172/*********************************************/
2173
2174USHORT
2175SiS_GetVCLK2Ptr(SiS_Private *SiS_Pr, USHORT ModeNo, USHORT ModeIdIndex,
2176 USHORT RefreshRateTableIndex, PSIS_HW_INFO HwInfo)
2177{
2178 USHORT CRT2Index,VCLKIndex=0,VCLKIndexGEN=0;
2179 USHORT modeflag,resinfo,tempbx;
2180 const UCHAR *CHTVVCLKPtr = NULL;
2181
2182 if(ModeNo <= 0x13) {
2183 modeflag = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_ModeFlag;
2184 resinfo = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_ResInfo;
2185 CRT2Index = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_CRT2CRTC;
2186 VCLKIndexGEN = (SiS_GetRegByte((SiS_Pr->SiS_P3ca+0x02)) >> 2) & 0x03;
2187 } else {
2188 modeflag = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_ModeFlag;
2189 resinfo = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_RESINFO;
2190 CRT2Index = SiS_Pr->SiS_RefIndex[RefreshRateTableIndex].Ext_CRT2CRTC;
2191 VCLKIndexGEN = SiS_Pr->SiS_RefIndex[RefreshRateTableIndex].Ext_CRTVCLK;
2192 if(HwInfo->jChipType < SIS_315H) VCLKIndexGEN &= 0x3f;
2193 }
2194
2195 if(SiS_Pr->SiS_VBType & VB_SISVB) { /* 30x/B/LV */
2196
2197 if(SiS_Pr->SiS_SetFlag & ProgrammingCRT2) {
2198
2199 CRT2Index >>= 6;
2200 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA)) { /* LCD */
2201
2202 if(HwInfo->jChipType < SIS_315H) {
2203 VCLKIndex = SiS_Pr->PanelVCLKIdx300;
2204 if((SiS_Pr->SiS_LCDInfo & DontExpandLCD) && (SiS_Pr->SiS_LCDInfo & LCDPass11)) {
2205 VCLKIndex = VCLKIndexGEN;
2206 }
2207 } else {
2208 VCLKIndex = SiS_Pr->PanelVCLKIdx315;
2209 if((SiS_Pr->SiS_LCDInfo & DontExpandLCD) && (SiS_Pr->SiS_LCDInfo & LCDPass11)) {
2210 switch(resinfo) {
2211 /* Only those whose IndexGEN doesn't match VBVCLK array */
2212 case SIS_RI_1280x720: VCLKIndex = VCLK_1280x720; break;
2213 case SIS_RI_720x480: VCLKIndex = VCLK_720x480; break;
2214 case SIS_RI_720x576: VCLKIndex = VCLK_720x576; break;
2215 case SIS_RI_768x576: VCLKIndex = VCLK_768x576; break;
2216 case SIS_RI_848x480: VCLKIndex = VCLK_848x480; break;
2217 case SIS_RI_856x480: VCLKIndex = VCLK_856x480; break;
2218 case SIS_RI_800x480: VCLKIndex = VCLK_800x480; break;
2219 case SIS_RI_1024x576: VCLKIndex = VCLK_1024x576; break;
2220 case SIS_RI_1152x864: VCLKIndex = VCLK_1152x864; break;
2221 case SIS_RI_1360x768: VCLKIndex = VCLK_1360x768; break;
2222 default: VCLKIndex = VCLKIndexGEN;
2223 }
2224
2225 if(ModeNo <= 0x13) {
2226 if(HwInfo->jChipType <= SIS_315PRO) {
2227 if(SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_CRT2CRTC == 1) VCLKIndex = 0x42;
2228 } else {
2229 if(SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_CRT2CRTC == 1) VCLKIndex = 0x00;
2230 }
2231 }
2232 if(HwInfo->jChipType <= SIS_315PRO) {
2233 if(VCLKIndex == 0) VCLKIndex = 0x41;
2234 if(VCLKIndex == 1) VCLKIndex = 0x43;
2235 if(VCLKIndex == 4) VCLKIndex = 0x44;
2236 }
2237 }
2238 }
2239
2240 } else if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) { /* TV */
2241
2242 if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) {
2243 if(SiS_Pr->SiS_TVMode & TVRPLLDIV2XO) VCLKIndex = HiTVVCLKDIV2;
2244 else VCLKIndex = HiTVVCLK;
2245 if(SiS_Pr->SiS_TVMode & TVSetTVSimuMode) {
2246 if(modeflag & Charx8Dot) VCLKIndex = HiTVSimuVCLK;
2247 else VCLKIndex = HiTVTextVCLK;
2248 }
2249 } else if(SiS_Pr->SiS_TVMode & TVSetYPbPr750p) VCLKIndex = YPbPr750pVCLK;
2250 else if(SiS_Pr->SiS_TVMode & TVSetYPbPr525p) VCLKIndex = TVVCLKDIV2;
2251 else if(SiS_Pr->SiS_TVMode & TVRPLLDIV2XO) VCLKIndex = TVVCLKDIV2;
2252 else VCLKIndex = TVVCLK;
2253
2254 if(HwInfo->jChipType < SIS_315H) VCLKIndex += TVCLKBASE_300;
2255 else VCLKIndex += TVCLKBASE_315;
2256
2257 } else { /* VGA2 */
2258
2259 VCLKIndex = VCLKIndexGEN;
2260 if(HwInfo->jChipType < SIS_315H) {
2261 if(ModeNo > 0x13) {
2262 if( (HwInfo->jChipType == SIS_630) &&
2263 (HwInfo->jChipRevision >= 0x30)) {
2264 if(VCLKIndex == 0x14) VCLKIndex = 0x34;
2265 }
2266 /* Better VGA2 clock for 1280x1024@75 */
2267 if(VCLKIndex == 0x17) VCLKIndex = 0x45;
2268 }
2269 }
2270 }
2271
2272 } else { /* If not programming CRT2 */
2273
2274 VCLKIndex = VCLKIndexGEN;
2275 if(HwInfo->jChipType < SIS_315H) {
2276 if(ModeNo > 0x13) {
2277 if( (HwInfo->jChipType != SIS_630) &&
2278 (HwInfo->jChipType != SIS_300) ) {
2279 if(VCLKIndex == 0x1b) VCLKIndex = 0x48;
2280 }
2281 }
2282 }
2283 }
2284
2285 } else { /* LVDS */
2286
2287 VCLKIndex = CRT2Index;
2288
2289 if(SiS_Pr->SiS_SetFlag & ProgrammingCRT2) {
2290
2291 if( (SiS_Pr->SiS_IF_DEF_CH70xx != 0) && (SiS_Pr->SiS_VBInfo & SetCRT2ToTV) ) {
2292
2293 VCLKIndex &= 0x1f;
2294 tempbx = 0;
2295 if(SiS_Pr->SiS_TVMode & TVSetCHOverScan) tempbx += 1;
2296 if(SiS_Pr->SiS_TVMode & TVSetPAL) {
2297 tempbx += 2;
2298 if(SiS_Pr->SiS_ModeType > ModeVGA) {
2299 if(SiS_Pr->SiS_CHSOverScan) tempbx = 8;
2300 }
2301 if(SiS_Pr->SiS_TVMode & TVSetPALM) {
2302 tempbx = 4;
2303 if(SiS_Pr->SiS_TVMode & TVSetCHOverScan) tempbx += 1;
2304 } else if(SiS_Pr->SiS_TVMode & TVSetPALN) {
2305 tempbx = 6;
2306 if(SiS_Pr->SiS_TVMode & TVSetCHOverScan) tempbx += 1;
2307 }
2308 }
2309 switch(tempbx) {
2310 case 0: CHTVVCLKPtr = SiS_Pr->SiS_CHTVVCLKUNTSC; break;
2311 case 1: CHTVVCLKPtr = SiS_Pr->SiS_CHTVVCLKONTSC; break;
2312 case 2: CHTVVCLKPtr = SiS_Pr->SiS_CHTVVCLKUPAL; break;
2313 case 3: CHTVVCLKPtr = SiS_Pr->SiS_CHTVVCLKOPAL; break;
2314 case 4: CHTVVCLKPtr = SiS_Pr->SiS_CHTVVCLKUPALM; break;
2315 case 5: CHTVVCLKPtr = SiS_Pr->SiS_CHTVVCLKOPALM; break;
2316 case 6: CHTVVCLKPtr = SiS_Pr->SiS_CHTVVCLKUPALN; break;
2317 case 7: CHTVVCLKPtr = SiS_Pr->SiS_CHTVVCLKOPALN; break;
2318 case 8: CHTVVCLKPtr = SiS_Pr->SiS_CHTVVCLKSOPAL; break;
2319 default: CHTVVCLKPtr = SiS_Pr->SiS_CHTVVCLKOPAL; break;
2320 }
2321 VCLKIndex = CHTVVCLKPtr[VCLKIndex];
2322
2323 } else if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) {
2324
2325 if(HwInfo->jChipType < SIS_315H) {
2326 VCLKIndex = SiS_Pr->PanelVCLKIdx300;
2327 } else {
2328 VCLKIndex = SiS_Pr->PanelVCLKIdx315;
2329 }
2330
2331 /* Special Timing: Barco iQ Pro R series */
2332 if(SiS_Pr->SiS_CustomT == CUT_BARCO1366) VCLKIndex = 0x44;
2333
2334 /* Special Timing: 848x480 parallel lvds */
2335 if(SiS_Pr->SiS_CustomT == CUT_PANEL848) {
2336 if(HwInfo->jChipType < SIS_315H) {
2337 VCLKIndex = VCLK34_300;
2338 /* if(resinfo == SIS_RI_1360x768) VCLKIndex = ?; */
2339 } else {
2340 VCLKIndex = VCLK34_315;
2341 /* if(resinfo == SIS_RI_1360x768) VCLKIndex = ?; */
2342 }
2343 }
2344
2345 } else {
2346
2347 VCLKIndex = VCLKIndexGEN;
2348 if(HwInfo->jChipType < SIS_315H) {
2349 if(ModeNo > 0x13) {
2350 if( (HwInfo->jChipType == SIS_630) &&
2351 (HwInfo->jChipRevision >= 0x30) ) {
2352 if(VCLKIndex == 0x14) VCLKIndex = 0x2e;
2353 }
2354 }
2355 }
2356 }
2357
2358 } else { /* if not programming CRT2 */
2359
2360 VCLKIndex = VCLKIndexGEN;
2361 if(HwInfo->jChipType < SIS_315H) {
2362 if(ModeNo > 0x13) {
2363 if( (HwInfo->jChipType != SIS_630) &&
2364 (HwInfo->jChipType != SIS_300) ) {
2365 if(VCLKIndex == 0x1b) VCLKIndex = 0x48;
2366 }
2367#if 0
2368 if(HwInfo->jChipType == SIS_730) {
2369 if(VCLKIndex == 0x0b) VCLKIndex = 0x40; /* 1024x768-70 */
2370 if(VCLKIndex == 0x0d) VCLKIndex = 0x41; /* 1024x768-75 */
2371 }
2372#endif
2373 }
2374 }
2375
2376 }
2377
2378 }
2379
2380#ifdef TWDEBUG
2381 xf86DrvMsg(0, X_INFO, "VCLKIndex %d (0x%x)\n", VCLKIndex, VCLKIndex);
2382#endif
2383
2384 return(VCLKIndex);
2385}
2386
2387/*********************************************/
2388/* SET CRT2 MODE TYPE REGISTERS */
2389/*********************************************/
2390
2391static void
2392SiS_SetCRT2ModeRegs(SiS_Private *SiS_Pr, USHORT ModeNo, USHORT ModeIdIndex,
2393 PSIS_HW_INFO HwInfo)
2394{
2395 USHORT i,j,modeflag;
2396 USHORT tempcl,tempah=0;
2397#if defined(SIS300) || defined(SIS315H)
2398 USHORT tempbl;
2399#endif
2400#ifdef SIS315H
2401 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
2402 USHORT tempah2, tempbl2;
2403#endif
2404
2405 if(ModeNo <= 0x13) {
2406 modeflag = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_ModeFlag;
2407 } else if(SiS_Pr->UseCustomMode) {
2408 modeflag = SiS_Pr->CModeFlag;
2409 } else {
2410 modeflag = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_ModeFlag;
2411 }
2412
2413 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA) {
2414
2415 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x00,0xAF,0x40);
2416 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x2E,0xF7);
2417
2418 } else {
2419
2420 for(i=0,j=4; i<3; i++,j++) SiS_SetReg(SiS_Pr->SiS_Part1Port,j,0);
2421 if(HwInfo->jChipType >= SIS_315H) {
2422 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x02,0x7F);
2423 }
2424
2425 tempcl = SiS_Pr->SiS_ModeType;
2426
2427 if(HwInfo->jChipType < SIS_315H) {
2428
2429#ifdef SIS300 /* ---- 300 series ---- */
2430
2431 /* For 301BDH: (with LCD via LVDS) */
2432 if(SiS_Pr->SiS_VBType & VB_NoLCD) {
2433 tempbl = SiS_GetReg(SiS_Pr->SiS_P3c4,0x32);
2434 tempbl &= 0xef;
2435 tempbl |= 0x02;
2436 if((SiS_Pr->SiS_VBInfo & SetCRT2ToTV) || (SiS_Pr->SiS_VBInfo & SetCRT2ToRAMDAC)) {
2437 tempbl |= 0x10;
2438 tempbl &= 0xfd;
2439 }
2440 SiS_SetReg(SiS_Pr->SiS_P3c4,0x32,tempbl);
2441 }
2442
2443 if(ModeNo > 0x13) {
2444 tempcl -= ModeVGA;
2445 if((tempcl > 0) || (tempcl == 0)) { /* tempcl is USHORT -> always true! */
2446 tempah = ((0x10 >> tempcl) | 0x80);
2447 }
2448 } else tempah = 0x80;
2449
2450 if(SiS_Pr->SiS_VBInfo & SetInSlaveMode) tempah ^= 0xA0;
2451
2452#endif /* SIS300 */
2453
2454 } else {
2455
2456#ifdef SIS315H /* ------- 315/330 series ------ */
2457
2458 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) {
2459 if(SiS_Pr->SiS_VBInfo & DisableCRT2Display) {
2460 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x2e,0x08);
2461 }
2462 }
2463
2464 if(ModeNo > 0x13) {
2465 tempcl -= ModeVGA;
2466 if((tempcl > 0) || (tempcl == 0)) { /* tempcl is USHORT -> always true! */
2467 tempah = (0x08 >> tempcl);
2468 if (tempah == 0) tempah = 1;
2469 tempah |= 0x40;
2470 }
2471 } else tempah = 0x40;
2472
2473 if(SiS_Pr->SiS_VBInfo & SetInSlaveMode) tempah ^= 0x50;
2474
2475#endif /* SIS315H */
2476
2477 }
2478
2479 if(SiS_Pr->SiS_VBInfo & DisableCRT2Display) tempah = 0;
2480
2481 if(HwInfo->jChipType < SIS_315H) {
2482 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x00,tempah);
2483 } else {
2484 if(SiS_Pr->SiS_IF_DEF_LVDS == 1) {
2485 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x00,0xa0,tempah);
2486 } else if(SiS_Pr->SiS_VBType & VB_SISVB) {
2487 if(IS_SIS740) {
2488 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x00,tempah);
2489 } else {
2490 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x00,0xa0,tempah);
2491 }
2492 }
2493 }
2494
2495 if(SiS_Pr->SiS_VBType & VB_SISVB) {
2496
2497 tempah = 0x01;
2498 if(!(SiS_Pr->SiS_VBInfo & SetInSlaveMode)) {
2499 tempah |= 0x02;
2500 }
2501 if(!(SiS_Pr->SiS_VBInfo & SetCRT2ToRAMDAC)) {
2502 tempah ^= 0x05;
2503 if(!(SiS_Pr->SiS_VBInfo & SetCRT2ToLCD)) {
2504 tempah ^= 0x01;
2505 }
2506 }
2507
2508 if(SiS_Pr->SiS_VBInfo & DisableCRT2Display) tempah = 0;
2509
2510 if(HwInfo->jChipType < SIS_315H) {
2511
2512 tempah = (tempah << 5) & 0xFF;
2513 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x01,tempah);
2514 tempah = (tempah >> 5) & 0xFF;
2515
2516 } else {
2517
2518 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x2E,0xF8,tempah);
2519
2520 }
2521
2522 if((SiS_Pr->SiS_ModeType == ModeVGA) && (!(SiS_Pr->SiS_VBInfo & SetInSlaveMode))) {
2523 tempah |= 0x10;
2524 }
2525
2526 tempah |= 0x80;
2527 if(SiS_Pr->SiS_VBType & VB_SIS301) {
2528 if(SiS_Pr->PanelXRes < 1280 && SiS_Pr->PanelYRes < 960) tempah &= ~0x80;
2529 }
2530
2531 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) {
2532 if(!(SiS_Pr->SiS_TVMode & (TVSetYPbPr750p | TVSetYPbPr525p))) {
2533 if(SiS_Pr->SiS_VBInfo & SetInSlaveMode) {
2534 tempah |= 0x20;
2535 }
2536 }
2537 }
2538
2539 SiS_SetRegANDOR(SiS_Pr->SiS_Part4Port,0x0D,0x40,tempah);
2540
2541 tempah = 0x80;
2542 if(SiS_Pr->SiS_VBType & VB_SIS301) {
2543 if(SiS_Pr->PanelXRes < 1280 && SiS_Pr->PanelYRes < 960) tempah = 0;
2544 }
2545
2546 if(SiS_IsDualLink(SiS_Pr, HwInfo)) tempah |= 0x40;
2547
2548 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) {
2549 if(SiS_Pr->SiS_TVMode & TVRPLLDIV2XO) {
2550 tempah |= 0x40;
2551 }
2552 }
2553
2554 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x0C,tempah);
2555
2556 } else { /* LVDS */
2557
2558 if(HwInfo->jChipType >= SIS_315H) {
2559
2560#ifdef SIS315H
2561 /* LVDS can only be slave in 8bpp modes */
2562 tempah = 0x80;
2563 if((modeflag & CRT2Mode) && (SiS_Pr->SiS_ModeType > ModeVGA)) {
2564 if(SiS_Pr->SiS_VBInfo & DriverMode) {
2565 tempah |= 0x02;
2566 }
2567 }
2568
2569 if(!(SiS_Pr->SiS_VBInfo & SetInSlaveMode)) {
2570 tempah |= 0x02;
2571 }
2572
2573 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) {
2574 tempah ^= 0x01;
2575 }
2576
2577 if(SiS_Pr->SiS_VBInfo & DisableCRT2Display) {
2578 tempah = 1;
2579 }
2580
2581 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x2e,0xF0,tempah);
2582#endif
2583
2584 } else {
2585
2586#ifdef SIS300
2587 tempah = 0;
2588 if( (!(SiS_Pr->SiS_VBInfo & SetInSlaveMode)) && (SiS_Pr->SiS_ModeType > ModeVGA) ) {
2589 tempah |= 0x02;
2590 }
2591 tempah <<= 5;
2592
2593 if(SiS_Pr->SiS_VBInfo & DisableCRT2Display) tempah = 0;
2594
2595 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x01,tempah);
2596#endif
2597
2598 }
2599
2600 }
2601
2602 } /* LCDA */
2603
2604 if(SiS_Pr->SiS_VBType & VB_SISVB) {
2605
2606 if(HwInfo->jChipType >= SIS_315H) {
2607
2608#ifdef SIS315H
2609 unsigned char bridgerev = SiS_GetReg(SiS_Pr->SiS_Part4Port,0x01);
2610
2611 /* The following is nearly unpreditable and varies from machine
2612 * to machine. Especially the 301DH seems to be a real trouble
2613 * maker. Some BIOSes simply set the registers (like in the
2614 * NoLCD-if-statements here), some set them according to the
2615 * LCDA stuff. It is very likely that some machines are not
2616 * treated correctly in the following, very case-orientated
2617 * code. What do I do then...?
2618 */
2619
2620 /* 740 variants match for 30xB, 301B-DH, 30xLV */
2621
2622 if(!(IS_SIS740)) {
2623 tempah = 0x04; /* For all bridges */
2624 tempbl = 0xfb;
2625 if(!(SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA)) {
2626 tempah = 0x00;
2627 if(SiS_IsDualEdge(SiS_Pr, HwInfo)) {
2628 tempbl = 0xff;
2629 }
2630 }
2631 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x13,tempbl,tempah);
2632 }
2633
2634 /* The following two are responsible for eventually wrong colors
2635 * in TV output. The DH (VB_NoLCD) conditions are unknown; the
2636 * b0 was found in some 651 machine (Pim; P4_23=0xe5); the b1 version
2637 * in a 650 box (Jake). What is the criteria?
2638 */
2639
2640 if((IS_SIS740) || (HwInfo->jChipType >= SIS_661) || (SiS_Pr->SiS_ROMNew)) {
2641 tempah = 0x30;
2642 tempbl = 0xc0;
2643 if((SiS_Pr->SiS_VBInfo & DisableCRT2Display) ||
2644 ((SiS_Pr->SiS_ROMNew) && (!(ROMAddr[0x5b] & 0x04)))) {
2645 tempah = 0x00;
2646 tempbl = 0x00;
2647 }
2648 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x2c,0xcf,tempah);
2649 SiS_SetRegANDOR(SiS_Pr->SiS_Part4Port,0x21,0x3f,tempbl);
2650 } else if(SiS_Pr->SiS_VBType & VB_SIS301) {
2651 /* Fixes "TV-blue-bug" on 315+301 */
2652 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x2c,0xcf); /* For 301 */
2653 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x21,0x3f);
2654 } else if(SiS_Pr->SiS_VBType & VB_SIS301LV302LV) {
2655 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x2c,0x30); /* For 30xLV */
2656 SiS_SetRegOR(SiS_Pr->SiS_Part4Port,0x21,0xc0);
2657 } else if((SiS_Pr->SiS_VBType & VB_NoLCD) && (bridgerev == 0xb0)) {
2658 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x2c,0x30); /* For 30xB-DH rev b0 (or "DH on 651"?) */
2659 SiS_SetRegOR(SiS_Pr->SiS_Part4Port,0x21,0xc0);
2660 } else {
2661 tempah = 0x30; tempah2 = 0xc0; /* For 30xB (and 301BDH rev b1) */
2662 tempbl = 0xcf; tempbl2 = 0x3f;
2663 if(!(SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA)) {
2664 tempah = tempah2 = 0x00;
2665 if(SiS_IsDualEdge(SiS_Pr, HwInfo)) {
2666 tempbl = tempbl2 = 0xff;
2667 }
2668 }
2669 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x2c,tempbl,tempah);
2670 SiS_SetRegANDOR(SiS_Pr->SiS_Part4Port,0x21,tempbl2,tempah2);
2671 }
2672
2673 if(IS_SIS740) {
2674 tempah = 0x80;
2675 if(SiS_Pr->SiS_VBInfo & DisableCRT2Display) tempah = 0x00;
2676 SiS_SetRegANDOR(SiS_Pr->SiS_Part4Port,0x23,0x7f,tempah);
2677 } else {
2678 tempah = 0x00;
2679 tempbl = 0x7f;
2680 if(!(SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA)) {
2681 tempbl = 0xff;
2682 if(!(SiS_IsDualEdge(SiS_Pr, HwInfo))) tempah = 0x80;
2683 }
2684 SiS_SetRegANDOR(SiS_Pr->SiS_Part4Port,0x23,tempbl,tempah);
2685 }
2686
2687#endif /* SIS315H */
2688
2689 } else if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) {
2690
2691#ifdef SIS300
2692 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x21,0x3f);
2693
2694 if((SiS_Pr->SiS_VBInfo & DisableCRT2Display) ||
2695 ((SiS_Pr->SiS_VBType & VB_NoLCD) &&
2696 (SiS_Pr->SiS_VBInfo & SetCRT2ToLCD))) {
2697 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x23,0x7F);
2698 } else {
2699 SiS_SetRegOR(SiS_Pr->SiS_Part4Port,0x23,0x80);
2700 }
2701#endif
2702
2703 }
2704
2705 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) {
2706 SiS_SetRegOR(SiS_Pr->SiS_Part4Port,0x0D,0x80);
2707 if(SiS_Pr->SiS_VBType & VB_SIS301C) {
2708 SiS_SetRegOR(SiS_Pr->SiS_Part4Port,0x3A,0xC0);
2709 }
2710 }
2711
2712 } else { /* LVDS */
2713
2714#ifdef SIS315H
2715 if(HwInfo->jChipType >= SIS_315H) {
2716
2717 if(SiS_Pr->SiS_IF_DEF_CH70xx != 0) {
2718
2719 tempah = 0x04;
2720 tempbl = 0xfb;
2721 if(!(SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA)) {
2722 tempah = 0x00;
2723 if(SiS_IsDualEdge(SiS_Pr, HwInfo)) tempbl = 0xff;
2724 }
2725 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x13,tempbl,tempah);
2726
2727 if(SiS_Pr->SiS_VBInfo & DisableCRT2Display) {
2728 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x13,0xfb);
2729 }
2730
2731 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x2c,0x30);
2732
2733 } else if(HwInfo->jChipType == SIS_550) {
2734
2735 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x13,0xfb);
2736 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x2c,0x30);
2737
2738 }
2739
2740 }
2741#endif
2742
2743 }
2744
2745}
2746
2747/*********************************************/
2748/* GET RESOLUTION DATA */
2749/*********************************************/
2750
2751USHORT
2752SiS_GetResInfo(SiS_Private *SiS_Pr,USHORT ModeNo,USHORT ModeIdIndex)
2753{
2754 if(ModeNo <= 0x13) return((USHORT)SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_ResInfo);
2755 else return((USHORT)SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_RESINFO);
2756}
2757
2758static void
2759SiS_GetCRT2ResInfo(SiS_Private *SiS_Pr,USHORT ModeNo,USHORT ModeIdIndex,
2760 PSIS_HW_INFO HwInfo)
2761{
2762 USHORT xres,yres,modeflag=0,resindex;
2763
2764 if(SiS_Pr->UseCustomMode) {
2765 xres = SiS_Pr->CHDisplay;
2766 if(SiS_Pr->CModeFlag & HalfDCLK) xres *= 2;
2767 SiS_Pr->SiS_VGAHDE = SiS_Pr->SiS_HDE = xres;
2768 yres = SiS_Pr->CVDisplay;
2769 if(SiS_Pr->CModeFlag & DoubleScanMode) yres *= 2;
2770 SiS_Pr->SiS_VGAVDE = SiS_Pr->SiS_VDE = yres;
2771 return;
2772 }
2773
2774 resindex = SiS_GetResInfo(SiS_Pr,ModeNo,ModeIdIndex);
2775
2776 if(ModeNo <= 0x13) {
2777 xres = SiS_Pr->SiS_StResInfo[resindex].HTotal;
2778 yres = SiS_Pr->SiS_StResInfo[resindex].VTotal;
2779 } else {
2780 xres = SiS_Pr->SiS_ModeResInfo[resindex].HTotal;
2781 yres = SiS_Pr->SiS_ModeResInfo[resindex].VTotal;
2782 modeflag = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_ModeFlag;
2783 }
2784
2785 if(!SiS_Pr->SiS_IF_DEF_DSTN && !SiS_Pr->SiS_IF_DEF_FSTN) {
2786
2787 if((HwInfo->jChipType >= SIS_315H) && (SiS_Pr->SiS_IF_DEF_LVDS == 1)) {
2788 if((ModeNo != 0x03) && (SiS_Pr->SiS_SetFlag & SetDOSMode)) {
2789 if(yres == 350) yres = 400;
2790 }
2791 if(SiS_GetReg(SiS_Pr->SiS_P3d4,0x3a) & 0x01) {
2792 if(ModeNo == 0x12) yres = 400;
2793 }
2794 }
2795
2796 if(modeflag & HalfDCLK) xres *= 2;
2797 if(modeflag & DoubleScanMode) yres *= 2;
2798
2799 }
2800
2801 if((SiS_Pr->SiS_VBType & VB_SISVB) && (!(SiS_Pr->SiS_VBType & VB_NoLCD))) {
2802
2803#if 0
2804 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToLCDA | SetCRT2ToLCD | SetCRT2ToHiVision)) {
2805 if(xres == 720) xres = 640;
2806 }
2807#endif
2808
2809 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) {
2810 switch(SiS_Pr->SiS_LCDResInfo) {
2811 case Panel_1024x768:
2812 if(!(SiS_Pr->SiS_SetFlag & LCDVESATiming)) {
2813 if(!(SiS_Pr->SiS_LCDInfo & DontExpandLCD)) {
2814 if(yres == 350) yres = 357;
2815 if(yres == 400) yres = 420;
2816 if(yres == 480) yres = 525;
2817 }
2818 }
2819 break;
2820 case Panel_1280x1024:
2821 if(!(SiS_Pr->SiS_LCDInfo & DontExpandLCD)) {
2822 /* BIOS bug - does this regardless of scaling */
2823 if(yres == 400) yres = 405;
2824 }
2825 if(yres == 350) yres = 360;
2826 if(SiS_Pr->SiS_SetFlag & LCDVESATiming) {
2827 if(yres == 360) yres = 375;
2828 }
2829 break;
2830 case Panel_1600x1200:
2831 if(!(SiS_Pr->SiS_SetFlag & LCDVESATiming)) {
2832 if(yres == 1024) yres = 1056;
2833 }
2834 break;
2835 }
2836 }
2837
2838 } else {
2839
2840 if(SiS_Pr->SiS_VBType & VB_SISVB) {
2841 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD | SetCRT2ToHiVision)) {
2842 if(xres == 720) xres = 640;
2843 }
2844 } else if(xres == 720) xres = 640;
2845
2846 if(SiS_Pr->SiS_SetFlag & SetDOSMode) {
2847 yres = 400;
2848 if(HwInfo->jChipType >= SIS_315H) {
2849 if(SiS_GetReg(SiS_Pr->SiS_P3c4,0x17) & 0x80) yres = 480;
2850 } else {
2851 if(SiS_GetReg(SiS_Pr->SiS_P3c4,0x13) & 0x80) yres = 480;
2852 }
2853 if(SiS_Pr->SiS_IF_DEF_DSTN || SiS_Pr->SiS_IF_DEF_FSTN) yres = 480;
2854 }
2855
2856 }
2857 SiS_Pr->SiS_VGAHDE = SiS_Pr->SiS_HDE = xres;
2858 SiS_Pr->SiS_VGAVDE = SiS_Pr->SiS_VDE = yres;
2859}
2860
2861/*********************************************/
2862/* GET CRT2 TIMING DATA */
2863/*********************************************/
2864
2865static BOOLEAN
2866SiS_GetLVDSCRT1Ptr(SiS_Private *SiS_Pr, USHORT ModeNo, USHORT ModeIdIndex,
2867 USHORT RefreshRateTableIndex, USHORT *ResIndex,
2868 USHORT *DisplayType)
2869 {
2870 USHORT modeflag=0;
2871
2872 if(SiS_Pr->SiS_IF_DEF_LVDS == 1) {
2873 if(!(SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA)) {
2874 if(!(SiS_Pr->SiS_VBInfo & SetInSlaveMode)) return FALSE;
2875 }
2876 } else if(SiS_Pr->SiS_VBType & VB_SISVB) {
2877 if(!(SiS_Pr->SiS_VBInfo & SetInSlaveMode)) return FALSE;
2878 } else
2879 return FALSE;
2880
2881 if(ModeNo <= 0x13) {
2882 modeflag = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_ModeFlag;
2883 (*ResIndex) = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_CRT2CRTC;
2884 } else {
2885 modeflag = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_ModeFlag;
2886 (*ResIndex) = SiS_Pr->SiS_RefIndex[RefreshRateTableIndex].Ext_CRT2CRTC;
2887 }
2888
2889 (*ResIndex) &= 0x3F;
2890
2891 if((SiS_Pr->SiS_IF_DEF_CH70xx) && (SiS_Pr->SiS_VBInfo & SetCRT2ToTV)) {
2892 (*DisplayType) = 18;
2893 if(SiS_Pr->SiS_TVMode & TVSetCHOverScan) (*DisplayType)++;
2894 if(SiS_Pr->SiS_TVMode & TVSetPAL) {
2895 (*DisplayType) += 2;
2896 if(SiS_Pr->SiS_ModeType > ModeVGA) {
2897 if(SiS_Pr->SiS_CHSOverScan) (*DisplayType) = 99;
2898 }
2899 if(SiS_Pr->SiS_TVMode & TVSetPALM) {
2900 (*DisplayType) = 18; /* PALM uses NTSC data */
2901 if(SiS_Pr->SiS_TVMode & TVSetCHOverScan) (*DisplayType)++;
2902 } else if(SiS_Pr->SiS_TVMode & TVSetPALN) {
2903 (*DisplayType) = 20; /* PALN uses PAL data */
2904 if(SiS_Pr->SiS_TVMode & TVSetCHOverScan) (*DisplayType)++;
2905 }
2906 }
2907 } else {
2908 switch(SiS_Pr->SiS_LCDResInfo) {
2909 case Panel_640x480: (*DisplayType) = 50; break;
2910 case Panel_640x480_2: (*DisplayType) = 52; break;
2911 case Panel_640x480_3: (*DisplayType) = 54; break;
2912 case Panel_800x600: (*DisplayType) = 0; break;
2913 case Panel_1024x600: (*DisplayType) = 23; break;
2914 case Panel_1024x768: (*DisplayType) = 4; break;
2915 case Panel_1152x768: (*DisplayType) = 27; break;
2916 case Panel_1280x768: (*DisplayType) = 40; break;
2917 case Panel_1280x1024: (*DisplayType) = 8; break;
2918 case Panel_1400x1050: (*DisplayType) = 14; break;
2919 case Panel_1600x1200: (*DisplayType) = 36; break;
2920 default: return FALSE;
2921 }
2922
2923 if(modeflag & HalfDCLK) (*DisplayType)++;
2924
2925 switch(SiS_Pr->SiS_LCDResInfo) {
2926 case Panel_640x480:
2927 case Panel_640x480_2:
2928 case Panel_640x480_3:
2929 break;
2930 default:
2931 if(SiS_Pr->SiS_LCDInfo & DontExpandLCD) (*DisplayType) += 2;
2932 }
2933
2934 if(SiS_Pr->SiS_LCDInfo & LCDPass11) {
2935 (*DisplayType) = 12;
2936 if(modeflag & HalfDCLK) (*DisplayType)++;
2937 }
2938 }
2939
2940#if 0
2941 if(SiS_Pr->SiS_IF_DEF_FSTN) {
2942 if(SiS_Pr->SiS_LCDResInfo == SiS_Pr->SiS_Panel320x480){
2943 (*DisplayType) = 22;
2944 }
2945 }
2946#endif
2947
2948 return TRUE;
2949}
2950
2951static void
2952SiS_GetCRT2Ptr(SiS_Private *SiS_Pr,USHORT ModeNo,USHORT ModeIdIndex,
2953 USHORT RefreshRateTableIndex,USHORT *CRT2Index,USHORT *ResIndex,
2954 PSIS_HW_INFO HwInfo)
2955{
2956 USHORT tempbx=0,tempal=0,resinfo=0;
2957
2958 if(ModeNo <= 0x13) {
2959 tempal = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_CRT2CRTC;
2960 } else {
2961 tempal = SiS_Pr->SiS_RefIndex[RefreshRateTableIndex].Ext_CRT2CRTC;
2962 resinfo = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_RESINFO;
2963 }
2964
2965 if((SiS_Pr->SiS_VBType & VB_SISVB) && (SiS_Pr->SiS_IF_DEF_LVDS == 0)) {
2966
2967 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) { /* LCD */
2968
2969 tempbx = SiS_Pr->SiS_LCDResInfo;
2970 if(!(SiS_Pr->SiS_SetFlag & LCDVESATiming)) tempbx += 32;
2971
2972 if(SiS_Pr->SiS_LCDResInfo == Panel_1680x1050) {
2973 if (resinfo == SIS_RI_1280x800) tempal = 9;
2974 else if(resinfo == SIS_RI_1400x1050) tempal = 11;
2975 } else if((SiS_Pr->SiS_LCDResInfo == Panel_1280x800) ||
2976 (SiS_Pr->SiS_LCDResInfo == Panel_1280x800_2)) {
2977 if (resinfo == SIS_RI_1280x768) tempal = 9;
2978 }
2979
2980 if(SiS_Pr->SiS_LCDInfo & DontExpandLCD) {
2981 /* Pass 1:1 only (center-screen handled outside) */
2982 /* This is never called for the panel's native resolution */
2983 /* since Pass1:1 will not be set in this case */
2984 tempbx = 100;
2985 if(ModeNo >= 0x13) {
2986 tempal = SiS_Pr->SiS_RefIndex[RefreshRateTableIndex].Ext_CRT2CRTC_NS;
2987 }
2988 }
2989
2990#ifdef SIS315H
2991 if(SiS_Pr->SiS_CustomT == CUT_COMPAQ1280) {
2992 if(SiS_Pr->SiS_LCDResInfo == Panel_1280x1024) {
2993 if(!(SiS_Pr->SiS_LCDInfo & DontExpandLCD)) {
2994 tempbx = 200;
2995 if(!(SiS_Pr->SiS_SetFlag & LCDVESATiming)) tempbx++;
2996 }
2997 }
2998 }
2999#endif
3000
3001 } else { /* TV */
3002
3003 if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) {
3004 /* if(SiS_Pr->SiS_VGAVDE > 480) SiS_Pr->SiS_TVMode &= (~TVSetTVSimuMode); */
3005 tempbx = 2;
3006 if(SiS_Pr->SiS_VBInfo & SetInSlaveMode) {
3007 tempbx = 13;
3008 if(!(SiS_Pr->SiS_TVMode & TVSetTVSimuMode)) tempbx = 14;
3009 }
3010 } else if(SiS_Pr->SiS_VBInfo & SetCRT2ToYPbPr525750) {
3011 if(SiS_Pr->SiS_TVMode & TVSetYPbPr750p) tempbx = 7;
3012 else if(SiS_Pr->SiS_TVMode & TVSetYPbPr525p) tempbx = 6;
3013 else tempbx = 5;
3014 if(SiS_Pr->SiS_TVMode & TVSetTVSimuMode) tempbx += 5;
3015 } else {
3016 if(SiS_Pr->SiS_TVMode & TVSetPAL) tempbx = 3;
3017 else tempbx = 4;
3018 if(SiS_Pr->SiS_TVMode & TVSetTVSimuMode) tempbx += 5;
3019 }
3020
3021 }
3022
3023 tempal &= 0x3F;
3024
3025 if(ModeNo > 0x13) {
3026 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTVNoHiVision) {
3027 if(tempal == 6) tempal = 7;
3028 if((resinfo == SIS_RI_720x480) ||
3029 (resinfo == SIS_RI_720x576) ||
3030 (resinfo == SIS_RI_768x576)) {
3031 tempal = 6;
3032 if(SiS_Pr->SiS_TVMode & (TVSetPAL | TVSetPALN)) {
3033 if(resinfo == SIS_RI_720x480) tempal = 9;
3034 }
3035 }
3036 if(SiS_Pr->SiS_VBInfo & SetCRT2ToYPbPr525750) {
3037 if(SiS_Pr->SiS_TVMode & TVSetYPbPr525p) {
3038 if(resinfo == SIS_RI_1024x768) tempal = 8;
3039 }
3040 if(SiS_Pr->SiS_TVMode & TVSetYPbPr750p) {
3041 if((resinfo == SIS_RI_720x576) ||
3042 (resinfo == SIS_RI_768x576)) {
3043 tempal = 8;
3044 }
3045 if(resinfo == SIS_RI_1280x720) tempal = 9;
3046 }
3047 }
3048 }
3049 }
3050
3051 *CRT2Index = tempbx;
3052 *ResIndex = tempal;
3053
3054 } else { /* LVDS, 301B-DH (if running on LCD) */
3055
3056 tempbx = 0;
3057 if((SiS_Pr->SiS_IF_DEF_CH70xx) && (SiS_Pr->SiS_VBInfo & SetCRT2ToTV)) {
3058
3059 tempbx = 10;
3060 if(SiS_Pr->SiS_TVMode & TVSetCHOverScan) tempbx += 1;
3061 if(SiS_Pr->SiS_TVMode & TVSetPAL) {
3062 tempbx += 2;
3063 if(SiS_Pr->SiS_ModeType > ModeVGA) {
3064 if(SiS_Pr->SiS_CHSOverScan) tempbx = 99;
3065 }
3066 if(SiS_Pr->SiS_TVMode & TVSetPALM) {
3067 tempbx = 90;
3068 if(SiS_Pr->SiS_TVMode & TVSetCHOverScan) tempbx += 1;
3069 } else if(SiS_Pr->SiS_TVMode & TVSetPALN) {
3070 tempbx = 92;
3071 if(SiS_Pr->SiS_TVMode & TVSetCHOverScan) tempbx += 1;
3072 }
3073 }
3074
3075 } else {
3076
3077 switch(SiS_Pr->SiS_LCDResInfo) {
3078 case Panel_640x480: tempbx = 6; break;
3079 case Panel_640x480_2:
3080 case Panel_640x480_3: tempbx = 30; break;
3081 case Panel_800x600: tempbx = 0; break;
3082 case Panel_1024x600: tempbx = 15; break;
3083 case Panel_1024x768: tempbx = 2; break;
3084 case Panel_1152x768: tempbx = 17; break;
3085 case Panel_1280x768: tempbx = 18; break;
3086 case Panel_1280x1024: tempbx = 4; break;
3087 case Panel_1400x1050: tempbx = 8; break;
3088 case Panel_1600x1200: tempbx = 21; break;
3089 case Panel_Barco1366: tempbx = 80; break;
3090 }
3091
3092 switch(SiS_Pr->SiS_LCDResInfo) {
3093 case Panel_640x480:
3094 case Panel_640x480_2:
3095 case Panel_640x480_3:
3096 break;
3097 default:
3098 if(SiS_Pr->SiS_LCDInfo & DontExpandLCD) tempbx++;
3099 }
3100
3101 if(SiS_Pr->SiS_LCDInfo & LCDPass11) tempbx = 7;
3102
3103 if(SiS_Pr->SiS_CustomT == CUT_BARCO1024) {
3104 tempbx = 82;
3105 if(SiS_Pr->SiS_LCDInfo & DontExpandLCD) tempbx++;
3106 } else if(SiS_Pr->SiS_CustomT == CUT_PANEL848) {
3107 tempbx = 84;
3108 if(SiS_Pr->SiS_LCDInfo & DontExpandLCD) tempbx++;
3109 }
3110
3111 if((SiS_Pr->SiS_CustomT != CUT_BARCO1366) &&
3112 (SiS_Pr->SiS_CustomT != CUT_PANEL848)) {
3113 if((SiS_Pr->SiS_LCDInfo & DontExpandLCD) &&
3114 (!(SiS_Pr->SiS_LCDInfo & LCDPass11))) {
3115 tempal = 0;
3116 }
3117 }
3118
3119 }
3120
3121 (*CRT2Index) = tempbx;
3122 (*ResIndex) = tempal & 0x1F;
3123 }
3124}
3125
3126static void
3127SiS_GetRAMDAC2DATA(SiS_Private *SiS_Pr,USHORT ModeNo,USHORT ModeIdIndex,
3128 USHORT RefreshRateTableIndex,PSIS_HW_INFO HwInfo)
3129{
3130 USHORT tempax=0,tempbx=0;
3131 USHORT temp1=0,modeflag=0,tempcx=0;
3132 USHORT index;
3133
3134 SiS_Pr->SiS_RVBHCMAX = 1;
3135 SiS_Pr->SiS_RVBHCFACT = 1;
3136
3137 if(ModeNo <= 0x13) {
3138
3139 modeflag = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_ModeFlag;
3140 index = SiS_GetModePtr(SiS_Pr,ModeNo,ModeIdIndex);
3141
3142 tempax = SiS_Pr->SiS_StandTable[index].CRTC[0];
3143 tempbx = SiS_Pr->SiS_StandTable[index].CRTC[6];
3144 temp1 = SiS_Pr->SiS_StandTable[index].CRTC[7];
3145
3146 } else {
3147
3148 modeflag = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_ModeFlag;
3149 index = SiS_Pr->SiS_RefIndex[RefreshRateTableIndex].Ext_CRT1CRTC;
3150
3151 tempax = SiS_Pr->SiS_CRT1Table[index].CR[0];
3152 tempax |= (SiS_Pr->SiS_CRT1Table[index].CR[14] << 8);
3153 tempax &= 0x03FF;
3154 tempbx = SiS_Pr->SiS_CRT1Table[index].CR[6];
3155 tempcx = SiS_Pr->SiS_CRT1Table[index].CR[13] << 8;
3156 tempcx &= 0x0100;
3157 tempcx <<= 2;
3158 tempbx |= tempcx;
3159 temp1 = SiS_Pr->SiS_CRT1Table[index].CR[7];
3160
3161 }
3162
3163 if(temp1 & 0x01) tempbx |= 0x0100;
3164 if(temp1 & 0x20) tempbx |= 0x0200;
3165
3166 tempax += 5;
3167
3168 /* Charx8Dot is no more used (and assumed), so we set it */
3169 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) {
3170 modeflag |= Charx8Dot;
3171 }
3172
3173 if(modeflag & Charx8Dot) tempax *= 8;
3174 else tempax *= 9;
3175
3176 if(modeflag & HalfDCLK) tempax <<= 1;
3177
3178 tempbx++;
3179
3180 SiS_Pr->SiS_VGAHT = SiS_Pr->SiS_HT = tempax;
3181 SiS_Pr->SiS_VGAVT = SiS_Pr->SiS_VT = tempbx;
3182}
3183
3184static void
3185SiS_GetCRT2DataLVDS(SiS_Private *SiS_Pr,USHORT ModeNo,USHORT ModeIdIndex,
3186 USHORT RefreshRateTableIndex, PSIS_HW_INFO HwInfo)
3187{
3188 USHORT CRT2Index, ResIndex;
3189 const SiS_LVDSDataStruct *LVDSData = NULL;
3190
3191 SiS_GetCRT2ResInfo(SiS_Pr, ModeNo, ModeIdIndex, HwInfo);
3192
3193 if(SiS_Pr->SiS_VBType & VB_SISVB) {
3194 SiS_Pr->SiS_RVBHCMAX = 1;
3195 SiS_Pr->SiS_RVBHCFACT = 1;
3196 SiS_Pr->SiS_NewFlickerMode = 0;
3197 SiS_Pr->SiS_RVBHRS = 50;
3198 SiS_Pr->SiS_RY1COE = 0;
3199 SiS_Pr->SiS_RY2COE = 0;
3200 SiS_Pr->SiS_RY3COE = 0;
3201 SiS_Pr->SiS_RY4COE = 0;
3202 }
3203
3204 if((SiS_Pr->SiS_VBType & VB_SISVB) && (SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA)) {
3205
3206#ifdef SIS315H
3207 if(SiS_Pr->SiS_LCDInfo & DontExpandLCD) {
3208 if(SiS_Pr->SiS_LCDInfo & LCDPass11) {
3209 if(SiS_Pr->UseCustomMode) {
3210 SiS_Pr->SiS_VGAHT = SiS_Pr->SiS_HT = SiS_Pr->CHTotal;
3211 SiS_Pr->SiS_VGAVT = SiS_Pr->SiS_VT = SiS_Pr->CVTotal;
3212 } else {
3213 if(ModeNo < 0x13) {
3214 ResIndex = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_CRT2CRTC;
3215 } else {
3216 ResIndex = SiS_Pr->SiS_RefIndex[RefreshRateTableIndex].Ext_CRT2CRTC_NS;
3217 }
3218 SiS_Pr->SiS_VGAHT = SiS_Pr->SiS_NoScaleData[ResIndex].VGAHT;
3219 SiS_Pr->SiS_VGAVT = SiS_Pr->SiS_NoScaleData[ResIndex].VGAVT;
3220 SiS_Pr->SiS_HT = SiS_Pr->SiS_NoScaleData[ResIndex].LCDHT;
3221 SiS_Pr->SiS_VT = SiS_Pr->SiS_NoScaleData[ResIndex].LCDVT;
3222 }
3223 } else {
3224 SiS_Pr->SiS_VGAHT = SiS_Pr->SiS_HT = SiS_Pr->PanelHT;
3225 SiS_Pr->SiS_VGAVT = SiS_Pr->SiS_VT = SiS_Pr->PanelVT;
3226 }
3227 } else {
3228 /* This handles custom modes and custom panels */
3229 SiS_Pr->SiS_HDE = SiS_Pr->PanelXRes;
3230 SiS_Pr->SiS_VDE = SiS_Pr->PanelYRes;
3231 SiS_Pr->SiS_HT = SiS_Pr->PanelHT;
3232 SiS_Pr->SiS_VT = SiS_Pr->PanelVT;
3233 SiS_Pr->SiS_VGAHT = SiS_Pr->PanelHT - (SiS_Pr->PanelXRes - SiS_Pr->SiS_VGAHDE);
3234 SiS_Pr->SiS_VGAVT = SiS_Pr->PanelVT - (SiS_Pr->PanelYRes - SiS_Pr->SiS_VGAVDE);
3235 }
3236
3237 SiS_CalcLCDACRT1Timing(SiS_Pr,ModeNo,ModeIdIndex);
3238
3239#endif
3240
3241 } else {
3242
3243 /* 301BDH needs LVDS Data */
3244 if((SiS_Pr->SiS_VBType & VB_NoLCD) && (SiS_Pr->SiS_VBInfo & SetCRT2ToLCD)) {
3245 SiS_Pr->SiS_IF_DEF_LVDS = 1;
3246 }
3247
3248 SiS_GetCRT2Ptr(SiS_Pr, ModeNo, ModeIdIndex, RefreshRateTableIndex,
3249 &CRT2Index, &ResIndex, HwInfo);
3250
3251 /* 301BDH needs LVDS Data */
3252 if((SiS_Pr->SiS_VBType & VB_NoLCD) && (SiS_Pr->SiS_VBInfo & SetCRT2ToLCD)) {
3253 SiS_Pr->SiS_IF_DEF_LVDS = 0;
3254 }
3255
3256 switch (CRT2Index) {
3257 case 0: LVDSData = SiS_Pr->SiS_LVDS800x600Data_1; break;
3258 case 1: LVDSData = SiS_Pr->SiS_LVDS800x600Data_2; break;
3259 case 2: LVDSData = SiS_Pr->SiS_LVDS1024x768Data_1; break;
3260 case 3: LVDSData = SiS_Pr->SiS_LVDS1024x768Data_2; break;
3261 case 4: LVDSData = SiS_Pr->SiS_LVDS1280x1024Data_1; break;
3262 case 5: LVDSData = SiS_Pr->SiS_LVDS1280x1024Data_2; break;
3263 case 6: LVDSData = SiS_Pr->SiS_LVDS640x480Data_1; break;
3264 case 7: LVDSData = SiS_Pr->SiS_LVDSXXXxXXXData_1; break;
3265 case 8: LVDSData = SiS_Pr->SiS_LVDS1400x1050Data_1; break;
3266 case 9: LVDSData = SiS_Pr->SiS_LVDS1400x1050Data_2; break;
3267 case 10: LVDSData = SiS_Pr->SiS_CHTVUNTSCData; break;
3268 case 11: LVDSData = SiS_Pr->SiS_CHTVONTSCData; break;
3269 case 12: LVDSData = SiS_Pr->SiS_CHTVUPALData; break;
3270 case 13: LVDSData = SiS_Pr->SiS_CHTVOPALData; break;
3271 case 14: LVDSData = SiS_Pr->SiS_LVDS320x480Data_1; break;
3272 case 15: LVDSData = SiS_Pr->SiS_LVDS1024x600Data_1; break;
3273 case 16: LVDSData = SiS_Pr->SiS_LVDS1024x600Data_2; break;
3274 case 17: LVDSData = SiS_Pr->SiS_LVDS1152x768Data_1; break;
3275 case 18: LVDSData = SiS_Pr->SiS_LVDS1152x768Data_2; break;
3276 case 19: LVDSData = SiS_Pr->SiS_LVDS1280x768Data_1; break;
3277 case 20: LVDSData = SiS_Pr->SiS_LVDS1280x768Data_2; break;
3278 case 21: LVDSData = SiS_Pr->SiS_LVDS1600x1200Data_1; break;
3279 case 22: LVDSData = SiS_Pr->SiS_LVDS1600x1200Data_2; break;
3280 case 30: LVDSData = SiS_Pr->SiS_LVDS640x480Data_2; break;
3281 case 80: LVDSData = SiS_Pr->SiS_LVDSBARCO1366Data_1; break;
3282 case 81: LVDSData = SiS_Pr->SiS_LVDSBARCO1366Data_2; break;
3283 case 82: LVDSData = SiS_Pr->SiS_LVDSBARCO1024Data_1; break;
3284 case 83: LVDSData = SiS_Pr->SiS_LVDSBARCO1024Data_2; break;
3285 case 84: LVDSData = SiS_Pr->SiS_LVDS848x480Data_1; break;
3286 case 85: LVDSData = SiS_Pr->SiS_LVDS848x480Data_2; break;
3287 case 90: LVDSData = SiS_Pr->SiS_CHTVUPALMData; break;
3288 case 91: LVDSData = SiS_Pr->SiS_CHTVOPALMData; break;
3289 case 92: LVDSData = SiS_Pr->SiS_CHTVUPALNData; break;
3290 case 93: LVDSData = SiS_Pr->SiS_CHTVOPALNData; break;
3291 case 99: LVDSData = SiS_Pr->SiS_CHTVSOPALData; break; /* Super Overscan */
3292 default: LVDSData = SiS_Pr->SiS_LVDS1024x768Data_1; break;
3293 }
3294
3295 SiS_Pr->SiS_VGAHT = (LVDSData+ResIndex)->VGAHT;
3296 SiS_Pr->SiS_VGAVT = (LVDSData+ResIndex)->VGAVT;
3297 SiS_Pr->SiS_HT = (LVDSData+ResIndex)->LCDHT;
3298 SiS_Pr->SiS_VT = (LVDSData+ResIndex)->LCDVT;
3299
3300 if(!(SiS_Pr->SiS_VBType & VB_SISVB)) {
3301 if((SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) && (!(SiS_Pr->SiS_LCDInfo & LCDPass11))) {
3302 if((!(SiS_Pr->SiS_LCDInfo & DontExpandLCD)) || (SiS_Pr->SiS_SetFlag & SetDOSMode)) {
3303 SiS_Pr->SiS_HDE = SiS_Pr->PanelXRes;
3304 SiS_Pr->SiS_VDE = SiS_Pr->PanelYRes;
3305 if(SiS_Pr->SiS_CustomT == CUT_BARCO1366) {
3306 if(ResIndex < 0x08) {
3307 SiS_Pr->SiS_HDE = 1280;
3308 SiS_Pr->SiS_VDE = 1024;
3309 }
3310 }
3311 }
3312 }
3313 }
3314 }
3315}
3316
3317static void
3318SiS_GetCRT2Data301(SiS_Private *SiS_Pr,USHORT ModeNo,USHORT ModeIdIndex,
3319 USHORT RefreshRateTableIndex,
3320 PSIS_HW_INFO HwInfo)
3321{
3322 UCHAR *ROMAddr = NULL;
3323 USHORT tempax,tempbx,modeflag,romptr=0;
3324 USHORT resinfo,CRT2Index,ResIndex;
3325 const SiS_LCDDataStruct *LCDPtr = NULL;
3326 const SiS_TVDataStruct *TVPtr = NULL;
3327#ifdef SIS315H
3328 SHORT resinfo661;
3329#endif
3330
3331 if(ModeNo <= 0x13) {
3332 modeflag = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_ModeFlag;
3333 resinfo = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_ResInfo;
3334 } else if(SiS_Pr->UseCustomMode) {
3335 modeflag = SiS_Pr->CModeFlag;
3336 resinfo = 0;
3337 } else {
3338 modeflag = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_ModeFlag;
3339 resinfo = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_RESINFO;
3340#ifdef SIS315H
3341 resinfo661 = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].ROMMODEIDX661;
3342 if( (SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) &&
3343 (SiS_Pr->SiS_SetFlag & LCDVESATiming) &&
3344 (resinfo661 >= 0) &&
3345 (SiS_Pr->SiS_NeedRomModeData) ) {
3346 if((ROMAddr = GetLCDStructPtr661(SiS_Pr, HwInfo))) {
3347 if((romptr = (SISGETROMW(21)))) {
3348 romptr += (resinfo661 * 10);
3349 ROMAddr = HwInfo->pjVirtualRomBase;
3350 }
3351 }
3352 }
3353#endif
3354 }
3355
3356 SiS_Pr->SiS_NewFlickerMode = 0;
3357 SiS_Pr->SiS_RVBHRS = 50;
3358 SiS_Pr->SiS_RY1COE = 0;
3359 SiS_Pr->SiS_RY2COE = 0;
3360 SiS_Pr->SiS_RY3COE = 0;
3361 SiS_Pr->SiS_RY4COE = 0;
3362
3363 SiS_GetCRT2ResInfo(SiS_Pr,ModeNo,ModeIdIndex,HwInfo);
3364
3365 if(SiS_Pr->SiS_VBInfo & SetCRT2ToRAMDAC){
3366
3367 if(SiS_Pr->UseCustomMode) {
3368
3369 SiS_Pr->SiS_RVBHCMAX = 1;
3370 SiS_Pr->SiS_RVBHCFACT = 1;
3371 SiS_Pr->SiS_VGAHT = SiS_Pr->CHTotal;
3372 SiS_Pr->SiS_VGAVT = SiS_Pr->CVTotal;
3373 SiS_Pr->SiS_HT = SiS_Pr->CHTotal;
3374 SiS_Pr->SiS_VT = SiS_Pr->CVTotal;
3375 SiS_Pr->SiS_HDE = SiS_Pr->SiS_VGAHDE;
3376 SiS_Pr->SiS_VDE = SiS_Pr->SiS_VGAVDE;
3377
3378 } else {
3379
3380 SiS_GetRAMDAC2DATA(SiS_Pr, ModeNo, ModeIdIndex, RefreshRateTableIndex, HwInfo);
3381
3382 }
3383
3384 } else if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) {
3385
3386 SiS_GetCRT2Ptr(SiS_Pr,ModeNo,ModeIdIndex,RefreshRateTableIndex,
3387 &CRT2Index,&ResIndex,HwInfo);
3388
3389 switch(CRT2Index) {
3390 case 2: TVPtr = SiS_Pr->SiS_ExtHiTVData; break;
3391 case 3: TVPtr = SiS_Pr->SiS_ExtPALData; break;
3392 case 4: TVPtr = SiS_Pr->SiS_ExtNTSCData; break;
3393 case 5: TVPtr = SiS_Pr->SiS_Ext525iData; break;
3394 case 6: TVPtr = SiS_Pr->SiS_Ext525pData; break;
3395 case 7: TVPtr = SiS_Pr->SiS_Ext750pData; break;
3396 case 8: TVPtr = SiS_Pr->SiS_StPALData; break;
3397 case 9: TVPtr = SiS_Pr->SiS_StNTSCData; break;
3398 case 10: TVPtr = SiS_Pr->SiS_St525iData; break;
3399 case 11: TVPtr = SiS_Pr->SiS_St525pData; break;
3400 case 12: TVPtr = SiS_Pr->SiS_St750pData; break;
3401 case 13: TVPtr = SiS_Pr->SiS_St1HiTVData; break;
3402 case 14: TVPtr = SiS_Pr->SiS_St2HiTVData; break;
3403 default: TVPtr = SiS_Pr->SiS_StPALData; break;
3404 }
3405
3406 SiS_Pr->SiS_RVBHCMAX = (TVPtr+ResIndex)->RVBHCMAX;
3407 SiS_Pr->SiS_RVBHCFACT = (TVPtr+ResIndex)->RVBHCFACT;
3408 SiS_Pr->SiS_VGAHT = (TVPtr+ResIndex)->VGAHT;
3409 SiS_Pr->SiS_VGAVT = (TVPtr+ResIndex)->VGAVT;
3410 SiS_Pr->SiS_HDE = (TVPtr+ResIndex)->TVHDE;
3411 SiS_Pr->SiS_VDE = (TVPtr+ResIndex)->TVVDE;
3412 SiS_Pr->SiS_RVBHRS = (TVPtr+ResIndex)->RVBHRS;
3413 SiS_Pr->SiS_NewFlickerMode = (TVPtr+ResIndex)->FlickerMode;
3414 if(modeflag & HalfDCLK) {
3415 SiS_Pr->SiS_RVBHRS = (TVPtr+ResIndex)->HALFRVBHRS;
3416 }
3417
3418 if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) {
3419
3420 if((resinfo == SIS_RI_1024x768) ||
3421 (resinfo == SIS_RI_1280x1024) ||
3422 (resinfo == SIS_RI_1280x720)) {
3423 SiS_Pr->SiS_NewFlickerMode = 0x40;
3424 }
3425
3426 if(SiS_Pr->SiS_VGAVDE == 350) SiS_Pr->SiS_TVMode |= TVSetTVSimuMode;
3427
3428 SiS_Pr->SiS_HT = ExtHiTVHT;
3429 SiS_Pr->SiS_VT = ExtHiTVVT;
3430 if(SiS_Pr->SiS_VBInfo & SetInSlaveMode) {
3431 if(SiS_Pr->SiS_TVMode & TVSetTVSimuMode) {
3432 SiS_Pr->SiS_HT = StHiTVHT;
3433 SiS_Pr->SiS_VT = StHiTVVT;
3434#if 0
3435 if(!(modeflag & Charx8Dot)) {
3436 SiS_Pr->SiS_HT = StHiTextTVHT;
3437 SiS_Pr->SiS_VT = StHiTextTVVT;
3438 }
3439#endif
3440 }
3441 }
3442
3443 } else if(SiS_Pr->SiS_VBInfo & SetCRT2ToYPbPr525750) {
3444
3445 if(SiS_Pr->SiS_TVMode & TVSetYPbPr750p) {
3446 SiS_Pr->SiS_HT = 1650;
3447 SiS_Pr->SiS_VT = 750;
3448 } else if(SiS_Pr->SiS_TVMode & TVSetYPbPr525p) {
3449 SiS_Pr->SiS_HT = NTSCHT;
3450 SiS_Pr->SiS_VT = NTSCVT;
3451 } else {
3452 SiS_Pr->SiS_HT = NTSCHT;
3453 if(SiS_Pr->SiS_TVMode & TVSetNTSC1024) SiS_Pr->SiS_HT = NTSC2HT;
3454 SiS_Pr->SiS_VT = NTSCVT;
3455 }
3456
3457 } else {
3458
3459 SiS_Pr->SiS_RY1COE = (TVPtr+ResIndex)->RY1COE;
3460 SiS_Pr->SiS_RY2COE = (TVPtr+ResIndex)->RY2COE;
3461 SiS_Pr->SiS_RY3COE = (TVPtr+ResIndex)->RY3COE;
3462 SiS_Pr->SiS_RY4COE = (TVPtr+ResIndex)->RY4COE;
3463
3464 if(modeflag & HalfDCLK) {
3465 SiS_Pr->SiS_RY1COE = 0x00;
3466 SiS_Pr->SiS_RY2COE = 0xf4;
3467 SiS_Pr->SiS_RY3COE = 0x10;
3468 SiS_Pr->SiS_RY4COE = 0x38;
3469 }
3470
3471 if(!(SiS_Pr->SiS_TVMode & TVSetPAL)) {
3472 SiS_Pr->SiS_HT = NTSCHT;
3473 if(SiS_Pr->SiS_TVMode & TVSetNTSC1024) SiS_Pr->SiS_HT = NTSC2HT;
3474 SiS_Pr->SiS_VT = NTSCVT;
3475 } else {
3476 SiS_Pr->SiS_HT = PALHT;
3477 SiS_Pr->SiS_VT = PALVT;
3478 }
3479
3480 }
3481
3482 } else if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) {
3483
3484 SiS_Pr->SiS_RVBHCMAX = 1;
3485 SiS_Pr->SiS_RVBHCFACT = 1;
3486
3487 if(SiS_Pr->UseCustomMode) {
3488
3489 SiS_Pr->SiS_VGAHT = SiS_Pr->CHTotal;
3490 SiS_Pr->SiS_VGAVT = SiS_Pr->CVTotal;
3491 SiS_Pr->SiS_HT = SiS_Pr->CHTotal;
3492 SiS_Pr->SiS_VT = SiS_Pr->CVTotal;
3493 SiS_Pr->SiS_HDE = SiS_Pr->SiS_VGAHDE;
3494 SiS_Pr->SiS_VDE = SiS_Pr->SiS_VGAVDE;
3495
3496 } else {
3497
3498 BOOLEAN gotit = FALSE;
3499
3500 if((SiS_Pr->SiS_LCDInfo & DontExpandLCD) && (!(SiS_Pr->SiS_LCDInfo & LCDPass11))) {
3501
3502 SiS_Pr->SiS_VGAHT = SiS_Pr->PanelHT;
3503 SiS_Pr->SiS_VGAVT = SiS_Pr->PanelVT;
3504 SiS_Pr->SiS_HT = SiS_Pr->PanelHT;
3505 SiS_Pr->SiS_VT = SiS_Pr->PanelVT;
3506 gotit = TRUE;
3507
3508 } else if( (!(SiS_Pr->SiS_LCDInfo & DontExpandLCD)) && (romptr) && (ROMAddr) ) {
3509
3510#ifdef SIS315H
3511 SiS_Pr->SiS_RVBHCMAX = ROMAddr[romptr];
3512 SiS_Pr->SiS_RVBHCFACT = ROMAddr[romptr+1];
3513 SiS_Pr->SiS_VGAHT = ROMAddr[romptr+2] | ((ROMAddr[romptr+3] & 0x0f) << 8);
3514 SiS_Pr->SiS_VGAVT = ROMAddr[romptr+4] | ((ROMAddr[romptr+3] & 0xf0) << 4);
3515 SiS_Pr->SiS_HT = ROMAddr[romptr+5] | ((ROMAddr[romptr+6] & 0x0f) << 8);
3516 SiS_Pr->SiS_VT = ROMAddr[romptr+7] | ((ROMAddr[romptr+6] & 0xf0) << 4);
3517 if(SiS_Pr->SiS_VGAHT) gotit = TRUE;
3518 else {
3519 SiS_Pr->SiS_LCDInfo |= DontExpandLCD;
3520 SiS_Pr->SiS_LCDInfo &= ~LCDPass11;
3521 SiS_Pr->SiS_VGAHT = SiS_Pr->PanelHT;
3522 SiS_Pr->SiS_VGAVT = SiS_Pr->PanelVT;
3523 SiS_Pr->SiS_HT = SiS_Pr->PanelHT;
3524 SiS_Pr->SiS_VT = SiS_Pr->PanelVT;
3525 gotit = TRUE;
3526 }
3527#endif
3528
3529 }
3530
3531 if(!gotit) {
3532
3533 SiS_GetCRT2Ptr(SiS_Pr,ModeNo,ModeIdIndex,RefreshRateTableIndex,
3534 &CRT2Index,&ResIndex,HwInfo);
3535
3536 switch(CRT2Index) {
3537 case Panel_1024x768 : LCDPtr = SiS_Pr->SiS_ExtLCD1024x768Data; break;
3538 case Panel_1024x768 + 32: LCDPtr = SiS_Pr->SiS_St2LCD1024x768Data; break;
3539 case Panel_1280x720 :
3540 case Panel_1280x720 + 32: LCDPtr = SiS_Pr->SiS_LCD1280x720Data; break;
3541 case Panel_1280x768_2 : LCDPtr = SiS_Pr->SiS_ExtLCD1280x768_2Data; break;
3542 case Panel_1280x768_2+ 32: LCDPtr = SiS_Pr->SiS_StLCD1280x768_2Data; break;
3543 case Panel_1280x800 :
3544 case Panel_1280x800 + 32: LCDPtr = SiS_Pr->SiS_LCD1280x800Data; break;
3545 case Panel_1280x800_2 :
3546 case Panel_1280x800_2+ 32: LCDPtr = SiS_Pr->SiS_LCD1280x800_2Data; break;
3547 case Panel_1280x960 :
3548 case Panel_1280x960 + 32: LCDPtr = SiS_Pr->SiS_LCD1280x960Data; break;
3549 case Panel_1280x1024 : LCDPtr = SiS_Pr->SiS_ExtLCD1280x1024Data; break;
3550 case Panel_1280x1024 + 32: LCDPtr = SiS_Pr->SiS_St2LCD1280x1024Data; break;
3551 case Panel_1400x1050 : LCDPtr = SiS_Pr->SiS_ExtLCD1400x1050Data; break;
3552 case Panel_1400x1050 + 32: LCDPtr = SiS_Pr->SiS_StLCD1400x1050Data; break;
3553 case Panel_1600x1200 : LCDPtr = SiS_Pr->SiS_ExtLCD1600x1200Data; break;
3554 case Panel_1600x1200 + 32: LCDPtr = SiS_Pr->SiS_StLCD1600x1200Data; break;
3555 case Panel_1680x1050 :
3556 case Panel_1680x1050 + 32: LCDPtr = SiS_Pr->SiS_LCD1680x1050Data; break;
3557 case 100 : LCDPtr = SiS_Pr->SiS_NoScaleData; break;
3558#ifdef SIS315H
3559 case 200 : LCDPtr = SiS310_ExtCompaq1280x1024Data; break;
3560 case 201 : LCDPtr = SiS_Pr->SiS_St2LCD1280x1024Data; break;
3561#endif
3562 default : LCDPtr = SiS_Pr->SiS_ExtLCD1024x768Data; break;
3563 }
3564
3565#ifdef TWDEBUG
3566 xf86DrvMsg(0, X_INFO, "GetCRT2Data: Index %d ResIndex %d\n", CRT2Index, ResIndex);
3567#endif
3568
3569 SiS_Pr->SiS_RVBHCMAX = (LCDPtr+ResIndex)->RVBHCMAX;
3570 SiS_Pr->SiS_RVBHCFACT = (LCDPtr+ResIndex)->RVBHCFACT;
3571 SiS_Pr->SiS_VGAHT = (LCDPtr+ResIndex)->VGAHT;
3572 SiS_Pr->SiS_VGAVT = (LCDPtr+ResIndex)->VGAVT;
3573 SiS_Pr->SiS_HT = (LCDPtr+ResIndex)->LCDHT;
3574 SiS_Pr->SiS_VT = (LCDPtr+ResIndex)->LCDVT;
3575
3576 }
3577
3578 tempax = SiS_Pr->PanelXRes;
3579 tempbx = SiS_Pr->PanelYRes;
3580
3581 if(SiS_Pr->SiS_LCDResInfo == Panel_1024x768) {
3582 if(SiS_Pr->SiS_SetFlag & LCDVESATiming) {
3583 if(HwInfo->jChipType < SIS_315H) {
3584 if (SiS_Pr->SiS_VGAVDE == 350) tempbx = 560;
3585 else if(SiS_Pr->SiS_VGAVDE == 400) tempbx = 640;
3586 }
3587 } else {
3588 if (SiS_Pr->SiS_VGAVDE == 357) tempbx = 527;
3589 else if(SiS_Pr->SiS_VGAVDE == 420) tempbx = 620;
3590 else if(SiS_Pr->SiS_VGAVDE == 525) tempbx = 775;
3591 else if(SiS_Pr->SiS_VGAVDE == 600) tempbx = 775;
3592 else if(SiS_Pr->SiS_VGAVDE == 350) tempbx = 560;
3593 else if(SiS_Pr->SiS_VGAVDE == 400) tempbx = 640;
3594 }
3595 } else if(SiS_Pr->SiS_LCDResInfo == Panel_1280x960) {
3596 if (SiS_Pr->SiS_VGAVDE == 350) tempbx = 700;
3597 else if(SiS_Pr->SiS_VGAVDE == 400) tempbx = 800;
3598 else if(SiS_Pr->SiS_VGAVDE == 1024) tempbx = 960;
3599 } else if(SiS_Pr->SiS_LCDResInfo == Panel_1280x1024) {
3600 if (SiS_Pr->SiS_VGAVDE == 360) tempbx = 768;
3601 else if(SiS_Pr->SiS_VGAVDE == 375) tempbx = 800;
3602 else if(SiS_Pr->SiS_VGAVDE == 405) tempbx = 864;
3603 } else if(SiS_Pr->SiS_LCDResInfo == Panel_1600x1200) {
3604 if(!(SiS_Pr->SiS_SetFlag & LCDVESATiming)) {
3605 if (SiS_Pr->SiS_VGAVDE == 350) tempbx = 875;
3606 else if(SiS_Pr->SiS_VGAVDE == 400) tempbx = 1000;
3607 }
3608 }
3609
3610 if(SiS_Pr->SiS_LCDInfo & DontExpandLCD) {
3611 tempax = SiS_Pr->SiS_VGAHDE;
3612 tempbx = SiS_Pr->SiS_VGAVDE;
3613 }
3614
3615 SiS_Pr->SiS_HDE = tempax;
3616 SiS_Pr->SiS_VDE = tempbx;
3617 }
3618 }
3619}
3620
3621static void
3622SiS_GetCRT2Data(SiS_Private *SiS_Pr, USHORT ModeNo, USHORT ModeIdIndex,
3623 USHORT RefreshRateTableIndex, PSIS_HW_INFO HwInfo)
3624{
3625
3626 if(SiS_Pr->SiS_VBType & VB_SISVB) {
3627
3628 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA) {
3629 SiS_GetCRT2DataLVDS(SiS_Pr, ModeNo, ModeIdIndex, RefreshRateTableIndex, HwInfo);
3630 } else {
3631 if((SiS_Pr->SiS_VBType & VB_NoLCD) && (SiS_Pr->SiS_VBInfo & SetCRT2ToLCD)) {
3632 /* Need LVDS Data for LCD on 301B-DH */
3633 SiS_GetCRT2DataLVDS(SiS_Pr, ModeNo, ModeIdIndex, RefreshRateTableIndex, HwInfo);
3634 } else {
3635 SiS_GetCRT2Data301(SiS_Pr, ModeNo, ModeIdIndex, RefreshRateTableIndex, HwInfo);
3636 }
3637 }
3638
3639 } else {
3640
3641 SiS_GetCRT2DataLVDS(SiS_Pr, ModeNo, ModeIdIndex, RefreshRateTableIndex, HwInfo);
3642
3643 }
3644}
3645
3646/*********************************************/
3647/* GET LVDS DES (SKEW) DATA */
3648/*********************************************/
3649
3650static void
3651SiS_GetLVDSDesPtr(SiS_Private *SiS_Pr, USHORT ModeNo, USHORT ModeIdIndex,
3652 USHORT RefreshRateTableIndex, USHORT *PanelIndex,
3653 USHORT *ResIndex, PSIS_HW_INFO HwInfo)
3654{
3655 USHORT modeflag;
3656
3657 if(ModeNo <= 0x13) {
3658 modeflag = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_ModeFlag;
3659 (*ResIndex) = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_CRT2CRTC;
3660 } else {
3661 modeflag = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_ModeFlag;
3662 (*ResIndex) = SiS_Pr->SiS_RefIndex[RefreshRateTableIndex].Ext_CRT2CRTC;
3663 }
3664
3665 (*ResIndex) &= 0x1F;
3666 (*PanelIndex) = 0;
3667
3668 if(SiS_Pr->SiS_IF_DEF_CH70xx != 0) {
3669 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) {
3670 (*PanelIndex) = 50;
3671 if((SiS_Pr->SiS_TVMode & TVSetPAL) && (!(SiS_Pr->SiS_TVMode & TVSetPALM))) (*PanelIndex) += 2;
3672 if(SiS_Pr->SiS_TVMode & TVSetCHOverScan) (*PanelIndex) += 1;
3673 /* Nothing special needed for SOverscan */
3674 /* PALM uses NTSC data, PALN uses PAL data */
3675 }
3676 }
3677
3678 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) {
3679 *PanelIndex = SiS_Pr->SiS_LCDTypeInfo;
3680 if(HwInfo->jChipType >= SIS_661) {
3681 /* As long as we don's use the BIOS tables, we
3682 * need to convert the TypeInfo as for 315 series
3683 */
3684 (*PanelIndex) = SiS_Pr->SiS_LCDResInfo - 1;
3685 }
3686 if(SiS_Pr->SiS_LCDInfo & DontExpandLCD) {
3687 (*PanelIndex) += 16;
3688 if(SiS_Pr->SiS_LCDInfo & LCDPass11) {
3689 (*PanelIndex) = 32;
3690 if(modeflag & HalfDCLK) (*PanelIndex)++;
3691 }
3692 }
3693 }
3694
3695 if(SiS_Pr->SiS_SetFlag & SetDOSMode) {
3696 if(SiS_Pr->SiS_LCDResInfo != Panel_640x480) {
3697 (*ResIndex) = 7;
3698 if(HwInfo->jChipType < SIS_315H) {
3699 if(SiS_GetReg(SiS_Pr->SiS_P3c4,0x13) & 0x80) (*ResIndex)++;
3700 }
3701 }
3702 }
3703}
3704
3705static void
3706SiS_GetLVDSDesData(SiS_Private *SiS_Pr, USHORT ModeNo,USHORT ModeIdIndex,
3707 USHORT RefreshRateTableIndex, PSIS_HW_INFO HwInfo)
3708{
3709 USHORT modeflag;
3710 USHORT PanelIndex,ResIndex;
3711 const SiS_LVDSDesStruct *PanelDesPtr = NULL;
3712
3713 SiS_Pr->SiS_LCDHDES = 0;
3714 SiS_Pr->SiS_LCDVDES = 0;
3715
3716 if( (SiS_Pr->UseCustomMode) ||
3717 (SiS_Pr->SiS_LCDResInfo == Panel_Custom) ||
3718 (SiS_Pr->SiS_CustomT == CUT_PANEL848) ||
3719 ((SiS_Pr->SiS_VBType & VB_SISVB) &&
3720 (SiS_Pr->SiS_LCDInfo & DontExpandLCD) &&
3721 (SiS_Pr->SiS_LCDInfo & LCDPass11)) ) {
3722 return;
3723 }
3724
3725 if((SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) && (SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA)) {
3726
3727#ifdef SIS315H
3728 if(SiS_Pr->SiS_LCDInfo & DontExpandLCD) {
3729 /* non-pass 1:1 only, see above */
3730 if(SiS_Pr->SiS_VGAHDE != SiS_Pr->PanelXRes) {
3731 SiS_Pr->SiS_LCDHDES = SiS_Pr->SiS_HT - ((SiS_Pr->PanelXRes - SiS_Pr->SiS_VGAHDE) / 2);
3732 }
3733 if(SiS_Pr->SiS_VGAVDE != SiS_Pr->PanelYRes) {
3734 SiS_Pr->SiS_LCDVDES = SiS_Pr->SiS_VT - ((SiS_Pr->PanelYRes - SiS_Pr->SiS_VGAVDE) / 2);
3735 }
3736 }
3737 if(SiS_Pr->SiS_VGAVDE == SiS_Pr->PanelYRes) {
3738 switch(SiS_Pr->SiS_CustomT) {
3739 case CUT_UNIWILL1024:
3740 case CUT_UNIWILL10242:
3741 case CUT_CLEVO1400:
3742 if(SiS_Pr->SiS_LCDResInfo == Panel_1024x768) {
3743 SiS_Pr->SiS_LCDVDES = SiS_Pr->PanelVT - 1;
3744 }
3745 break;
3746 }
3747 if(SiS_Pr->SiS_LCDResInfo == Panel_1280x1024) {
3748 if(SiS_Pr->SiS_CustomT != CUT_COMPAQ1280) {
3749 SiS_Pr->SiS_LCDVDES = SiS_Pr->PanelVT - 1;
3750 }
3751 }
3752 }
3753#endif
3754
3755 } else {
3756
3757 SiS_GetLVDSDesPtr(SiS_Pr, ModeNo, ModeIdIndex, RefreshRateTableIndex,
3758 &PanelIndex, &ResIndex, HwInfo);
3759
3760 switch(PanelIndex) {
3761 case 0: PanelDesPtr = SiS_Pr->SiS_PanelType00_1; break; /* --- */
3762 case 1: PanelDesPtr = SiS_Pr->SiS_PanelType01_1; break;
3763 case 2: PanelDesPtr = SiS_Pr->SiS_PanelType02_1; break;
3764 case 3: PanelDesPtr = SiS_Pr->SiS_PanelType03_1; break;
3765 case 4: PanelDesPtr = SiS_Pr->SiS_PanelType04_1; break;
3766 case 5: PanelDesPtr = SiS_Pr->SiS_PanelType05_1; break;
3767 case 6: PanelDesPtr = SiS_Pr->SiS_PanelType06_1; break;
3768 case 7: PanelDesPtr = SiS_Pr->SiS_PanelType07_1; break;
3769 case 8: PanelDesPtr = SiS_Pr->SiS_PanelType08_1; break;
3770 case 9: PanelDesPtr = SiS_Pr->SiS_PanelType09_1; break;
3771 case 10: PanelDesPtr = SiS_Pr->SiS_PanelType0a_1; break;
3772 case 11: PanelDesPtr = SiS_Pr->SiS_PanelType0b_1; break;
3773 case 12: PanelDesPtr = SiS_Pr->SiS_PanelType0c_1; break;
3774 case 13: PanelDesPtr = SiS_Pr->SiS_PanelType0d_1; break;
3775 case 14: PanelDesPtr = SiS_Pr->SiS_PanelType0e_1; break;
3776 case 15: PanelDesPtr = SiS_Pr->SiS_PanelType0f_1; break;
3777 case 16: PanelDesPtr = SiS_Pr->SiS_PanelType00_2; break; /* --- */
3778 case 17: PanelDesPtr = SiS_Pr->SiS_PanelType01_2; break;
3779 case 18: PanelDesPtr = SiS_Pr->SiS_PanelType02_2; break;
3780 case 19: PanelDesPtr = SiS_Pr->SiS_PanelType03_2; break;
3781 case 20: PanelDesPtr = SiS_Pr->SiS_PanelType04_2; break;
3782 case 21: PanelDesPtr = SiS_Pr->SiS_PanelType05_2; break;
3783 case 22: PanelDesPtr = SiS_Pr->SiS_PanelType06_2; break;
3784 case 23: PanelDesPtr = SiS_Pr->SiS_PanelType07_2; break;
3785 case 24: PanelDesPtr = SiS_Pr->SiS_PanelType08_2; break;
3786 case 25: PanelDesPtr = SiS_Pr->SiS_PanelType09_2; break;
3787 case 26: PanelDesPtr = SiS_Pr->SiS_PanelType0a_2; break;
3788 case 27: PanelDesPtr = SiS_Pr->SiS_PanelType0b_2; break;
3789 case 28: PanelDesPtr = SiS_Pr->SiS_PanelType0c_2; break;
3790 case 29: PanelDesPtr = SiS_Pr->SiS_PanelType0d_2; break;
3791 case 30: PanelDesPtr = SiS_Pr->SiS_PanelType0e_2; break;
3792 case 31: PanelDesPtr = SiS_Pr->SiS_PanelType0f_2; break;
3793 case 32: PanelDesPtr = SiS_Pr->SiS_PanelTypeNS_1; break; /* pass 1:1 */
3794 case 33: PanelDesPtr = SiS_Pr->SiS_PanelTypeNS_2; break;
3795 case 50: PanelDesPtr = SiS_Pr->SiS_CHTVUNTSCDesData; break; /* TV */
3796 case 51: PanelDesPtr = SiS_Pr->SiS_CHTVONTSCDesData; break;
3797 case 52: PanelDesPtr = SiS_Pr->SiS_CHTVUPALDesData; break;
3798 case 53: PanelDesPtr = SiS_Pr->SiS_CHTVOPALDesData; break;
3799 default: return;
3800 }
3801
3802 SiS_Pr->SiS_LCDHDES = (PanelDesPtr+ResIndex)->LCDHDES;
3803 SiS_Pr->SiS_LCDVDES = (PanelDesPtr+ResIndex)->LCDVDES;
3804
3805 if((ModeNo <= 0x13) && (SiS_Pr->SiS_LCDInfo & DontExpandLCD)) {
3806 modeflag = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_ModeFlag;
3807 if((SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) && (SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA)) {
3808 if(!(modeflag & HalfDCLK)) SiS_Pr->SiS_LCDHDES = 632;
3809 } else if(!(SiS_Pr->SiS_SetFlag & SetDOSMode)) {
3810 if(SiS_Pr->SiS_LCDResInfo != Panel_1280x1024) {
3811 if(SiS_Pr->SiS_LCDResInfo >= Panel_1024x768) {
3812 if(HwInfo->jChipType < SIS_315H) {
3813 if(!(modeflag & HalfDCLK)) SiS_Pr->SiS_LCDHDES = 320;
3814 } else {
3815 if(SiS_Pr->SiS_LCDResInfo == Panel_1024x768) SiS_Pr->SiS_LCDHDES = 480;
3816 if(SiS_Pr->SiS_LCDResInfo == Panel_1400x1050) SiS_Pr->SiS_LCDHDES = 804;
3817 if(SiS_Pr->SiS_LCDResInfo == Panel_1600x1200) SiS_Pr->SiS_LCDHDES = 704;
3818 if(!(modeflag & HalfDCLK)) {
3819 SiS_Pr->SiS_LCDHDES = 320;
3820 if(SiS_Pr->SiS_LCDResInfo == Panel_1400x1050) SiS_Pr->SiS_LCDHDES = 632;
3821 if(SiS_Pr->SiS_LCDResInfo == Panel_1600x1200) SiS_Pr->SiS_LCDHDES = 542;
3822 }
3823 }
3824 }
3825 }
3826 }
3827 }
3828 }
3829}
3830
3831/*********************************************/
3832/* DISABLE VIDEO BRIDGE */
3833/*********************************************/
3834
3835/* NEVER use any variables (VBInfo), this will be called
3836 * from outside the context of modeswitch!
3837 * MUST call getVBType before calling this
3838 */
3839void
3840SiS_DisableBridge(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
3841{
3842#ifdef SIS315H
3843 USHORT tempah,pushax=0,modenum;
3844#endif
3845 USHORT temp=0;
3846
3847 if(SiS_Pr->SiS_VBType & VB_SISVB) {
3848
3849 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) { /* ===== For 30xB/LV ===== */
3850
3851 if(HwInfo->jChipType < SIS_315H) {
3852
3853#ifdef SIS300 /* 300 series */
3854
3855 if(!(SiS_CR36BIOSWord23b(SiS_Pr,HwInfo))) {
3856 if(SiS_Pr->SiS_VBType & VB_SIS301LV302LV) {
3857 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x26,0xFE);
3858 } else {
3859 SiS_SetRegSR11ANDOR(SiS_Pr,HwInfo,0xF7,0x08);
3860 }
3861 SiS_PanelDelay(SiS_Pr, HwInfo, 3);
3862 }
3863 if(SiS_Is301B(SiS_Pr)) {
3864 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x1f,0x3f);
3865 SiS_ShortDelay(SiS_Pr,1);
3866 }
3867 SiS_SetRegAND(SiS_Pr->SiS_Part2Port,0x00,0xDF);
3868 SiS_DisplayOff(SiS_Pr);
3869 SiS_SetRegAND(SiS_Pr->SiS_P3c4,0x32,0xDF);
3870 SiS_SetRegAND(SiS_Pr->SiS_P3c4,0x1E,0xDF);
3871 SiS_UnLockCRT2(SiS_Pr,HwInfo);
3872 if(!(SiS_Pr->SiS_VBType & VB_SIS301LV302LV)) {
3873 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x01,0x80);
3874 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x02,0x40);
3875 }
3876 if( (!(SiS_CRT2IsLCD(SiS_Pr, HwInfo))) ||
3877 (!(SiS_CR36BIOSWord23d(SiS_Pr, HwInfo))) ) {
3878 SiS_PanelDelay(SiS_Pr, HwInfo, 2);
3879 if(SiS_Pr->SiS_VBType & VB_SIS301LV302LV) {
3880 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x26,0xFD);
3881 } else {
3882 SiS_SetRegSR11ANDOR(SiS_Pr,HwInfo,0xFB,0x04);
3883 }
3884 }
3885
3886#endif /* SIS300 */
3887
3888 } else {
3889
3890#ifdef SIS315H /* 315 series */
3891
3892 BOOLEAN custom1 = ((SiS_Pr->SiS_CustomT == CUT_COMPAQ1280) ||
3893 (SiS_Pr->SiS_CustomT == CUT_CLEVO1400)) ? TRUE : FALSE;
3894
3895 modenum = SiS_GetReg(SiS_Pr->SiS_P3d4,0x34) & 0x7f;
3896
3897 if(SiS_Pr->SiS_VBType & VB_SIS301LV302LV) {
3898
3899#ifdef SET_EMI
3900 if(SiS_Pr->SiS_VBType & (VB_SIS302LV | VB_SIS302ELV)) {
3901 if(SiS_Pr->SiS_CustomT != CUT_CLEVO1400) {
3902 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x30,0x0c);
3903 }
3904 }
3905#endif
3906 if( (modenum <= 0x13) ||
3907 (SiS_IsVAMode(SiS_Pr,HwInfo)) ||
3908 (!(SiS_IsDualEdge(SiS_Pr,HwInfo))) ) {
3909 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x26,0xFE);
3910 if(custom1) SiS_PanelDelay(SiS_Pr, HwInfo, 3);
3911 }
3912
3913 if(!custom1) {
3914 SiS_DDC2Delay(SiS_Pr,0xff00);
3915 SiS_DDC2Delay(SiS_Pr,0xe000);
3916 SiS_SetRegByte(SiS_Pr->SiS_P3c6,0x00);
3917 pushax = SiS_GetReg(SiS_Pr->SiS_P3c4,0x06);
3918 if(IS_SIS740) {
3919 SiS_SetRegAND(SiS_Pr->SiS_P3c4,0x06,0xE3);
3920 }
3921 SiS_PanelDelay(SiS_Pr, HwInfo, 3);
3922 }
3923
3924 }
3925
3926 if(!(SiS_IsNotM650orLater(SiS_Pr, HwInfo))) {
3927 if(HwInfo->jChipType < SIS_340) {
3928 tempah = 0xef;
3929 if(SiS_IsVAMode(SiS_Pr,HwInfo)) tempah = 0xf7;
3930 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x4c,tempah);
3931 }
3932 }
3933
3934 if(SiS_Pr->SiS_VBType & VB_SIS301LV302LV) {
3935 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x1F,~0x10);
3936 }
3937
3938 tempah = 0x3f;
3939 if(SiS_IsDualEdge(SiS_Pr,HwInfo)) {
3940 tempah = 0x7f;
3941 if(!(SiS_IsVAMode(SiS_Pr,HwInfo))) tempah = 0xbf;
3942 }
3943 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x1F,tempah);
3944
3945 if((SiS_IsVAMode(SiS_Pr,HwInfo)) ||
3946 ((SiS_Pr->SiS_VBType & VB_SIS301LV302LV) && (modenum <= 0x13))) {
3947
3948 SiS_DisplayOff(SiS_Pr);
3949 if(SiS_Pr->SiS_VBType & VB_SIS301LV302LV) {
3950 SiS_PanelDelay(SiS_Pr, HwInfo, 2);
3951 }
3952 SiS_SetRegAND(SiS_Pr->SiS_P3c4,0x32,0xDF);
3953 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x1E,0xDF);
3954
3955 }
3956
3957 if((!(SiS_IsVAMode(SiS_Pr,HwInfo))) ||
3958 ((SiS_Pr->SiS_VBType & VB_SIS301LV302LV) && (modenum <= 0x13))) {
3959
3960 if(!(SiS_IsDualEdge(SiS_Pr,HwInfo))) {
3961 SiS_SetRegAND(SiS_Pr->SiS_Part2Port,0x00,0xdf);
3962 SiS_DisplayOff(SiS_Pr);
3963 }
3964 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x00,0x80);
3965
3966 if(SiS_Pr->SiS_VBType & VB_SIS301LV302LV) {
3967 SiS_PanelDelay(SiS_Pr, HwInfo, 2);
3968 }
3969
3970 SiS_SetRegAND(SiS_Pr->SiS_P3c4,0x32,0xDF);
3971 temp = SiS_GetReg(SiS_Pr->SiS_Part1Port,0x00);
3972 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x00,0x10);
3973 SiS_SetRegAND(SiS_Pr->SiS_P3c4,0x1E,0xDF);
3974 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x00,temp);
3975
3976 }
3977
3978 if(SiS_IsNotM650orLater(SiS_Pr,HwInfo)) {
3979 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x2e,0x7f);
3980 }
3981
3982 if(SiS_Pr->SiS_VBType & VB_SIS301LV302LV) {
3983
3984 if(!custom1) {
3985
3986 if(!(SiS_IsVAMode(SiS_Pr,HwInfo))) {
3987 if(!(SiS_CRT2IsLCD(SiS_Pr,HwInfo))) {
3988 if(!(SiS_IsDualEdge(SiS_Pr,HwInfo))) {
3989 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x26,0xFD);
3990 }
3991 }
3992 }
3993
3994 SiS_SetReg(SiS_Pr->SiS_P3c4,0x06,pushax);
3995
3996 if(SiS_Pr->SiS_VBType & (VB_SIS302LV | VB_SIS302ELV)) {
3997 if(SiS_IsVAorLCD(SiS_Pr, HwInfo)) {
3998 SiS_PanelDelayLoop(SiS_Pr, HwInfo, 3, 20);
3999 }
4000 }
4001
4002 } else {
4003
4004 if((SiS_IsVAMode(SiS_Pr,HwInfo)) ||
4005 (!(SiS_IsDualEdge(SiS_Pr,HwInfo)))) {
4006 if((!(SiS_WeHaveBacklightCtrl(SiS_Pr, HwInfo))) ||
4007 (!(SiS_CRT2IsLCD(SiS_Pr, HwInfo)))) {
4008 SiS_PanelDelay(SiS_Pr, HwInfo, 2);
4009 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x26,0xFD);
4010 SiS_PanelDelay(SiS_Pr, HwInfo, 4);
4011 }
4012 }
4013
4014 }
4015 }
4016
4017#endif /* SIS315H */
4018
4019 }
4020
4021 } else { /* ============ For 301 ================ */
4022
4023 if(HwInfo->jChipType < SIS_315H) {
4024#ifdef SIS300
4025 if(!(SiS_CR36BIOSWord23b(SiS_Pr,HwInfo))) {
4026 SiS_SetRegSR11ANDOR(SiS_Pr,HwInfo,0xF7,0x08);
4027 SiS_PanelDelay(SiS_Pr, HwInfo, 3);
4028 }
4029#endif
4030 }
4031
4032 SiS_SetRegAND(SiS_Pr->SiS_Part2Port,0x00,0xDF); /* disable VB */
4033 SiS_DisplayOff(SiS_Pr);
4034
4035 if(HwInfo->jChipType >= SIS_315H) {
4036 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x00,0x80);
4037 }
4038
4039 SiS_SetRegAND(SiS_Pr->SiS_P3c4,0x32,0xDF); /* disable lock mode */
4040
4041 if(HwInfo->jChipType >= SIS_315H) {
4042 temp = SiS_GetReg(SiS_Pr->SiS_Part1Port,0x00);
4043 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x00,0x10);
4044 SiS_SetRegOR(SiS_Pr->SiS_P3c4,0x1E,0x20);
4045 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x00,temp);
4046 } else {
4047#ifdef SIS300
4048 SiS_SetRegAND(SiS_Pr->SiS_P3c4,0x1E,0xDF); /* disable CRT2 */
4049 if( (!(SiS_CRT2IsLCD(SiS_Pr, HwInfo))) ||
4050 (!(SiS_CR36BIOSWord23d(SiS_Pr,HwInfo))) ) {
4051 SiS_PanelDelay(SiS_Pr, HwInfo, 2);
4052 SiS_SetRegSR11ANDOR(SiS_Pr,HwInfo,0xFB,0x04);
4053 }
4054#endif
4055 }
4056
4057 }
4058
4059 } else { /* ============ For LVDS =============*/
4060
4061 if(HwInfo->jChipType < SIS_315H) {
4062
4063#ifdef SIS300 /* 300 series */
4064
4065 if(SiS_Pr->SiS_IF_DEF_CH70xx == 1) {
4066 SiS_SetCH700x(SiS_Pr,0x090E);
4067 }
4068
4069 if(HwInfo->jChipType == SIS_730) {
4070 if(!(SiS_GetReg(SiS_Pr->SiS_P3c4,0x11) & 0x08)) {
4071 SiS_WaitVBRetrace(SiS_Pr,HwInfo);
4072 }
4073 if(!(SiS_CR36BIOSWord23b(SiS_Pr,HwInfo))) {
4074 SiS_SetRegSR11ANDOR(SiS_Pr,HwInfo,0xF7,0x08);
4075 SiS_PanelDelay(SiS_Pr, HwInfo, 3);
4076 }
4077 } else {
4078 if(!(SiS_GetReg(SiS_Pr->SiS_P3c4,0x11) & 0x08)) {
4079 if(!(SiS_GetReg(SiS_Pr->SiS_P3c4,0x13) & 0x40)) {
4080 if(!(SiS_CR36BIOSWord23b(SiS_Pr,HwInfo))) {
4081 SiS_WaitVBRetrace(SiS_Pr,HwInfo);
4082 if(!(SiS_GetReg(SiS_Pr->SiS_P3c4,0x06) & 0x1c)) {
4083 SiS_DisplayOff(SiS_Pr);
4084 }
4085 SiS_SetRegSR11ANDOR(SiS_Pr,HwInfo,0xF7,0x08);
4086 SiS_PanelDelay(SiS_Pr, HwInfo, 3);
4087 }
4088 }
4089 }
4090 }
4091
4092 SiS_DisplayOff(SiS_Pr);
4093
4094 SiS_SetRegAND(SiS_Pr->SiS_P3c4,0x32,0xDF);
4095
4096 SiS_SetRegAND(SiS_Pr->SiS_P3c4,0x1E,0xDF);
4097 SiS_UnLockCRT2(SiS_Pr,HwInfo);
4098 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x01,0x80);
4099 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x02,0x40);
4100
4101 if( (!(SiS_CRT2IsLCD(SiS_Pr, HwInfo))) ||
4102 (!(SiS_CR36BIOSWord23d(SiS_Pr,HwInfo))) ) {
4103 SiS_PanelDelay(SiS_Pr, HwInfo, 2);
4104 SiS_SetRegSR11ANDOR(SiS_Pr,HwInfo,0xFB,0x04);
4105 }
4106
4107#endif /* SIS300 */
4108
4109 } else {
4110
4111#ifdef SIS315H /* 315 series */
4112
4113 if(!(SiS_IsNotM650orLater(SiS_Pr,HwInfo))) {
4114 if(HwInfo->jChipType < SIS_340) {
4115 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x4c,~0x18);
4116 }
4117 }
4118
4119 if(SiS_Pr->SiS_IF_DEF_CH70xx != 0) {
4120
4121 if(HwInfo->jChipType == SIS_740) {
4122 temp = SiS_GetCH701x(SiS_Pr,0x61);
4123 if(temp < 1) {
4124 SiS_SetCH701x(SiS_Pr,0xac76);
4125 SiS_SetCH701x(SiS_Pr,0x0066);
4126 }
4127
4128 if( (!(SiS_IsDualEdge(SiS_Pr,HwInfo))) ||
4129 (SiS_IsTVOrYPbPrOrScart(SiS_Pr,HwInfo)) ) {
4130 SiS_SetCH701x(SiS_Pr,0x3e49);
4131 }
4132 }
4133
4134 if( (!(SiS_IsDualEdge(SiS_Pr,HwInfo))) ||
4135 (SiS_IsVAMode(SiS_Pr,HwInfo)) ) {
4136 SiS_Chrontel701xBLOff(SiS_Pr);
4137 SiS_Chrontel701xOff(SiS_Pr,HwInfo);
4138 }
4139
4140 if(HwInfo->jChipType != SIS_740) {
4141 if( (!(SiS_IsDualEdge(SiS_Pr,HwInfo))) ||
4142 (SiS_IsTVOrYPbPrOrScart(SiS_Pr,HwInfo)) ) {
4143 SiS_SetCH701x(SiS_Pr,0x0149);
4144 }
4145 }
4146
4147 }
4148
4149 if(SiS_Pr->SiS_IF_DEF_CH70xx == 0) {
4150 SiS_SetRegSR11ANDOR(SiS_Pr,HwInfo,0xF7,0x08);
4151 SiS_PanelDelay(SiS_Pr, HwInfo, 3);
4152 }
4153
4154 if( (SiS_Pr->SiS_IF_DEF_CH70xx == 0) ||
4155 (!(SiS_IsDualEdge(SiS_Pr,HwInfo))) ||
4156 (!(SiS_IsTVOrYPbPrOrScart(SiS_Pr,HwInfo))) ) {
4157 SiS_DisplayOff(SiS_Pr);
4158 }
4159
4160 if( (SiS_Pr->SiS_IF_DEF_CH70xx == 0) ||
4161 (!(SiS_IsDualEdge(SiS_Pr,HwInfo))) ||
4162 (!(SiS_IsVAMode(SiS_Pr,HwInfo))) ) {
4163 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x00,0x80);
4164 }
4165
4166 if(HwInfo->jChipType == SIS_740) {
4167 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x2e,0x7f);
4168 }
4169
4170 SiS_SetRegAND(SiS_Pr->SiS_P3c4,0x32,0xDF);
4171
4172 if( (SiS_Pr->SiS_IF_DEF_CH70xx == 0) ||
4173 (!(SiS_IsDualEdge(SiS_Pr,HwInfo))) ||
4174 (!(SiS_IsVAMode(SiS_Pr,HwInfo))) ) {
4175 SiS_SetRegAND(SiS_Pr->SiS_P3c4,0x1E,0xDF);
4176 }
4177
4178 if(SiS_Pr->SiS_IF_DEF_CH70xx == 0) {
4179 if(SiS_CRT2IsLCD(SiS_Pr, HwInfo)) {
4180 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x1e,0xdf);
4181 if(HwInfo->jChipType == SIS_550) {
4182 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x1e,0xbf);
4183 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x1e,0xef);
4184 }
4185 }
4186 } else {
4187 if(HwInfo->jChipType == SIS_740) {
4188 if(SiS_IsLCDOrLCDA(SiS_Pr,HwInfo)) {
4189 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x1e,0xdf);
4190 }
4191 } else if(SiS_IsVAMode(SiS_Pr,HwInfo)) {
4192 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x1e,0xdf);
4193 }
4194 }
4195
4196 if(SiS_Pr->SiS_IF_DEF_CH70xx != 0) {
4197 if(SiS_IsDualEdge(SiS_Pr,HwInfo)) {
4198 /* SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x13,0xff); */
4199 } else {
4200 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x13,0xfb);
4201 }
4202 }
4203
4204 SiS_UnLockCRT2(SiS_Pr,HwInfo);
4205
4206 if(HwInfo->jChipType == SIS_550) {
4207 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x01,0x80); /* DirectDVD PAL?*/
4208 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x02,0x40); /* VB clock / 4 ? */
4209 } else if( (SiS_Pr->SiS_IF_DEF_CH70xx == 0) ||
4210 (!(SiS_IsDualEdge(SiS_Pr,HwInfo))) ||
4211 (!(SiS_IsVAMode(SiS_Pr,HwInfo))) ) {
4212 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x2e,0xf7);
4213 }
4214
4215 if(SiS_Pr->SiS_IF_DEF_CH70xx == 0) {
4216 if(SiS_CRT2IsLCD(SiS_Pr, HwInfo)) {
4217 if(!(SiS_WeHaveBacklightCtrl(SiS_Pr,HwInfo))) {
4218 SiS_PanelDelay(SiS_Pr, HwInfo, 2);
4219 SiS_SetRegSR11ANDOR(SiS_Pr,HwInfo,0xFB,0x04);
4220 }
4221 }
4222 }
4223
4224#endif /* SIS315H */
4225
4226 } /* 315 series */
4227
4228 } /* LVDS */
4229
4230}
4231
4232/*********************************************/
4233/* ENABLE VIDEO BRIDGE */
4234/*********************************************/
4235
4236/* NEVER use any variables (VBInfo), this will be called
4237 * from outside the context of a mode switch!
4238 * MUST call getVBType before calling this
4239 */
Adrian Bunk75c96f82005-05-05 16:16:09 -07004240static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07004241SiS_EnableBridge(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
4242{
4243 USHORT temp=0,tempah;
4244#ifdef SIS315H
4245 USHORT temp1,pushax=0;
4246 BOOLEAN delaylong = FALSE;
4247#endif
4248
4249 if(SiS_Pr->SiS_VBType & VB_SISVB) {
4250
4251 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) { /* ====== For 301B et al ====== */
4252
4253 if(HwInfo->jChipType < SIS_315H) {
4254
4255#ifdef SIS300 /* 300 series */
4256
4257 if(SiS_CRT2IsLCD(SiS_Pr, HwInfo)) {
4258 if(SiS_Pr->SiS_VBType & VB_SIS301LV302LV) {
4259 SiS_SetRegOR(SiS_Pr->SiS_Part4Port,0x26,0x02);
4260 } else if(SiS_Pr->SiS_VBType & VB_NoLCD) {
4261 SiS_SetRegSR11ANDOR(SiS_Pr,HwInfo,0xFB,0x00);
4262 }
4263 if(SiS_Pr->SiS_VBType & (VB_SIS301LV302LV | VB_NoLCD)) {
4264 if(!(SiS_CR36BIOSWord23d(SiS_Pr, HwInfo))) {
4265 SiS_PanelDelay(SiS_Pr, HwInfo, 0);
4266 }
4267 }
4268 }
4269
4270 if((SiS_Pr->SiS_VBType & VB_NoLCD) &&
4271 (SiS_CRT2IsLCD(SiS_Pr, HwInfo))) {
4272
4273 SiS_SetRegOR(SiS_Pr->SiS_P3c4,0x1E,0x20); /* Enable CRT2 */
4274 SiS_DisplayOn(SiS_Pr);
4275 SiS_UnLockCRT2(SiS_Pr,HwInfo);
4276 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x02,0xBF);
4277 if(SiS_BridgeInSlavemode(SiS_Pr)) {
4278 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x01,0x1F);
4279 } else {
4280 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x01,0x1F,0x40);
4281 }
4282 if(!(SiS_GetReg(SiS_Pr->SiS_P3c4,0x13) & 0x40)) {
4283 if(!(SiS_GetReg(SiS_Pr->SiS_P3c4,0x16) & 0x10)) {
4284 if(!(SiS_CR36BIOSWord23b(SiS_Pr,HwInfo))) {
4285 SiS_PanelDelay(SiS_Pr, HwInfo, 1);
4286 }
4287 SiS_WaitVBRetrace(SiS_Pr,HwInfo);
4288 SiS_SetRegSR11ANDOR(SiS_Pr,HwInfo,0xF7,0x00);
4289 }
4290 }
4291
4292 } else {
4293
4294 temp = SiS_GetReg(SiS_Pr->SiS_P3c4,0x32) & 0xDF; /* lock mode */
4295 if(SiS_BridgeInSlavemode(SiS_Pr)) {
4296 tempah = SiS_GetReg(SiS_Pr->SiS_P3d4,0x30);
4297 if(!(tempah & SetCRT2ToRAMDAC)) temp |= 0x20;
4298 }
4299 SiS_SetReg(SiS_Pr->SiS_P3c4,0x32,temp);
4300 SiS_SetRegOR(SiS_Pr->SiS_P3c4,0x1E,0x20);
4301 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x00,0x1F,0x20); /* enable VB processor */
4302 SiS_SetRegOR(SiS_Pr->SiS_Part4Port,0x1F,0xC0);
4303 SiS_DisplayOn(SiS_Pr);
4304 if(SiS_Pr->SiS_VBType & VB_SIS301LV302LV) {
4305 if(SiS_CRT2IsLCD(SiS_Pr, HwInfo)) {
4306 if(!(SiS_GetReg(SiS_Pr->SiS_P3c4,0x16) & 0x10)) {
4307 if(!(SiS_CR36BIOSWord23b(SiS_Pr,HwInfo))) {
4308 SiS_PanelDelay(SiS_Pr, HwInfo, 1);
4309 }
4310 SiS_SetRegOR(SiS_Pr->SiS_Part4Port,0x26,0x01);
4311 }
4312 }
4313 }
4314
4315 }
4316
4317
4318#endif /* SIS300 */
4319
4320 } else {
4321
4322#ifdef SIS315H /* 315 series */
4323
4324#ifdef SET_EMI
4325 UCHAR r30=0, r31=0, r32=0, r33=0, cr36=0;
4326 /* USHORT emidelay=0; */
4327#endif
4328
4329 if(SiS_Pr->SiS_VBType & VB_SIS301LV302LV) {
4330 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x1f,0xef);
4331#ifdef SET_EMI
4332 if(SiS_Pr->SiS_VBType & (VB_SIS302LV | VB_SIS302ELV)) {
4333 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x30,0x0c);
4334 }
4335#endif
4336 }
4337
4338 if(!(SiS_IsNotM650orLater(SiS_Pr, HwInfo))) {
4339 if(HwInfo->jChipType < SIS_340) {
4340 tempah = 0x10;
4341 if(SiS_LCDAEnabled(SiS_Pr, HwInfo)) {
4342 if(SiS_TVEnabled(SiS_Pr, HwInfo)) tempah = 0x18;
4343 else tempah = 0x08;
4344 }
4345 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x4c,tempah);
4346 }
4347 }
4348
4349 if(SiS_Pr->SiS_VBType & VB_SIS301LV302LV) {
4350
4351 SiS_SetRegByte(SiS_Pr->SiS_P3c6,0x00);
4352 SiS_DisplayOff(SiS_Pr);
4353 pushax = SiS_GetReg(SiS_Pr->SiS_P3c4,0x06);
4354 if(IS_SIS740) {
4355 SiS_SetRegAND(SiS_Pr->SiS_P3c4,0x06,0xE3);
4356 }
4357
4358 if(SiS_IsVAorLCD(SiS_Pr, HwInfo)) {
4359 if(!(SiS_GetReg(SiS_Pr->SiS_Part4Port,0x26) & 0x02)) {
4360 SiS_PanelDelayLoop(SiS_Pr, HwInfo, 3, 2);
4361 SiS_SetRegOR(SiS_Pr->SiS_Part4Port,0x26,0x02);
4362 SiS_PanelDelayLoop(SiS_Pr, HwInfo, 3, 2);
4363 if(SiS_Pr->SiS_VBType & (VB_SIS302LV | VB_SIS302ELV)) {
4364 SiS_GenericDelay(SiS_Pr, 0x4500);
4365 }
4366 }
4367 }
4368
4369 if(!(SiS_GetReg(SiS_Pr->SiS_P3d4,0x31) & 0x40)) {
4370 SiS_PanelDelayLoop(SiS_Pr, HwInfo, 3, 10);
4371 delaylong = TRUE;
4372 }
4373
4374 }
4375
4376 if(!(SiS_IsVAMode(SiS_Pr,HwInfo))) {
4377
4378 temp = SiS_GetReg(SiS_Pr->SiS_P3c4,0x32) & 0xDF;
4379 if(SiS_BridgeInSlavemode(SiS_Pr)) {
4380 tempah = SiS_GetReg(SiS_Pr->SiS_P3d4,0x30);
4381 if(!(tempah & SetCRT2ToRAMDAC)) {
4382 if(!(SiS_LCDAEnabled(SiS_Pr, HwInfo))) temp |= 0x20;
4383 }
4384 }
4385 SiS_SetReg(SiS_Pr->SiS_P3c4,0x32,temp);
4386
4387 SiS_SetRegOR(SiS_Pr->SiS_P3c4,0x1E,0x20); /* enable CRT2 */
4388
4389 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x2e,0x7f);
4390 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x2e,0x80);
4391
4392 if(SiS_Pr->SiS_VBType & VB_SIS301LV302LV) {
4393 SiS_PanelDelay(SiS_Pr, HwInfo, 2);
4394 }
4395
4396 } else {
4397
4398 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x1e,0x20);
4399
4400 }
4401
4402 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x00,0x1f,0x20);
4403 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x2e,0x80);
4404
4405 tempah = 0xc0;
4406 if(SiS_IsDualEdge(SiS_Pr, HwInfo)) {
4407 tempah = 0x80;
4408 if(!(SiS_IsVAMode(SiS_Pr, HwInfo))) tempah = 0x40;
4409 }
4410 SiS_SetRegOR(SiS_Pr->SiS_Part4Port,0x1F,tempah);
4411
4412 if(SiS_Pr->SiS_VBType & VB_SIS301LV302LV) {
4413
4414 SiS_PanelDelay(SiS_Pr, HwInfo, 2);
4415
4416 SiS_SetRegOR(SiS_Pr->SiS_Part4Port,0x1f,0x10);
4417 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x2e,0x80);
4418
4419 if(SiS_Pr->SiS_CustomT != CUT_CLEVO1400) {
4420#ifdef SET_EMI
4421 if(SiS_Pr->SiS_VBType & (VB_SIS302LV | VB_SIS302ELV)) {
4422 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x30,0x0c);
4423 SiS_GenericDelay(SiS_Pr, 0x500);
4424 }
4425#endif
4426 SiS_SetRegOR(SiS_Pr->SiS_Part4Port,0x27,0x0c);
4427
4428 if(SiS_Pr->SiS_VBType & (VB_SIS302LV | VB_SIS302ELV)) {
4429#ifdef SET_EMI
4430 cr36 = SiS_GetReg(SiS_Pr->SiS_P3d4,0x36);
4431
4432 if(SiS_Pr->SiS_ROMNew) {
4433 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
4434 USHORT romptr = GetLCDStructPtr661_2(SiS_Pr, HwInfo);
4435 if(romptr) {
4436 SiS_SetRegOR(SiS_Pr->SiS_Part4Port,0x30,0x20); /* Reset */
4437 SiS_Pr->EMI_30 = 0;
4438 SiS_Pr->EMI_31 = ROMAddr[romptr + SiS_Pr->SiS_EMIOffset + 0];
4439 SiS_Pr->EMI_32 = ROMAddr[romptr + SiS_Pr->SiS_EMIOffset + 1];
4440 SiS_Pr->EMI_33 = ROMAddr[romptr + SiS_Pr->SiS_EMIOffset + 2];
4441 if(ROMAddr[romptr + 1] & 0x10) SiS_Pr->EMI_30 = 0x40;
4442 /* emidelay = SISGETROMW((romptr + 0x22)); */
4443 SiS_Pr->HaveEMI = SiS_Pr->HaveEMILCD = SiS_Pr->OverruleEMI = TRUE;
4444 }
4445 }
4446
4447 /* (P4_30|0x40) */
4448 /* Compal 1400x1050: 0x05, 0x60, 0x00 YES (1.10.7w; CR36=69) */
4449 /* Compal 1400x1050: 0x0d, 0x70, 0x40 YES (1.10.7x; CR36=69) */
4450 /* Acer 1280x1024: 0x12, 0xd0, 0x6b NO (1.10.9k; CR36=73) */
4451 /* Compaq 1280x1024: 0x0d, 0x70, 0x6b YES (1.12.04b; CR36=03) */
4452 /* Clevo 1024x768: 0x05, 0x60, 0x33 NO (1.10.8e; CR36=12, DL!) */
4453 /* Clevo 1024x768: 0x0d, 0x70, 0x40 (if type == 3) YES (1.10.8y; CR36=?2) */
4454 /* Clevo 1024x768: 0x05, 0x60, 0x33 (if type != 3) YES (1.10.8y; CR36=?2) */
4455 /* Asus 1024x768: ? ? (1.10.8o; CR36=?2) */
4456 /* Asus 1024x768: 0x08, 0x10, 0x3c (problematic) YES (1.10.8q; CR36=22) */
4457
4458 if(SiS_Pr->HaveEMI) {
4459 r30 = SiS_Pr->EMI_30; r31 = SiS_Pr->EMI_31;
4460 r32 = SiS_Pr->EMI_32; r33 = SiS_Pr->EMI_33;
4461 } else {
4462 r30 = 0;
4463 }
4464
4465 /* EMI_30 is read at driver start; however, the BIOS sets this
4466 * (if it is used) only if the LCD is in use. In case we caught
4467 * the machine while on TV output, this bit is not set and we
4468 * don't know if it should be set - hence our detection is wrong.
4469 * Work-around this here:
4470 */
4471
4472 if((!SiS_Pr->HaveEMI) || (!SiS_Pr->HaveEMILCD)) {
4473 switch((cr36 & 0x0f)) {
4474 case 2:
4475 r30 |= 0x40;
4476 if(SiS_Pr->SiS_CustomT == CUT_CLEVO1024) r30 &= ~0x40;
4477 if(!SiS_Pr->HaveEMI) {
4478 r31 = 0x05; r32 = 0x60; r33 = 0x33;
4479 if((cr36 & 0xf0) == 0x30) {
4480 r31 = 0x0d; r32 = 0x70; r33 = 0x40;
4481 }
4482 }
4483 break;
4484 case 3: /* 1280x1024 */
4485 if(SiS_Pr->SiS_CustomT == CUT_COMPAQ1280) r30 |= 0x40;
4486 if(!SiS_Pr->HaveEMI) {
4487 r31 = 0x12; r32 = 0xd0; r33 = 0x6b;
4488 if(SiS_Pr->SiS_CustomT == CUT_COMPAQ1280) {
4489 r31 = 0x0d; r32 = 0x70; r33 = 0x6b;
4490 }
4491 }
4492 break;
4493 case 9: /* 1400x1050 */
4494 r30 |= 0x40;
4495 if(!SiS_Pr->HaveEMI) {
4496 r31 = 0x05; r32 = 0x60; r33 = 0x00;
4497 if(SiS_Pr->SiS_CustomT == CUT_COMPAL1400_2) {
4498 r31 = 0x0d; r32 = 0x70; r33 = 0x40; /* BIOS values */
4499 }
4500 }
4501 break;
4502 case 11: /* 1600x1200 - unknown */
4503 r30 |= 0x40;
4504 if(!SiS_Pr->HaveEMI) {
4505 r31 = 0x05; r32 = 0x60; r33 = 0x00;
4506 }
4507 }
4508 }
4509
4510 /* BIOS values don't work so well sometimes */
4511 if(!SiS_Pr->OverruleEMI) {
4512#ifdef COMPAL_HACK
4513 if(SiS_Pr->SiS_CustomT == CUT_COMPAL1400_2) {
4514 if((cr36 & 0x0f) == 0x09) {
4515 r30 = 0x60; r31 = 0x05; r32 = 0x60; r33 = 0x00;
4516 }
4517 }
4518#endif
4519#ifdef COMPAQ_HACK
4520 if(SiS_Pr->SiS_CustomT == CUT_COMPAQ1280) {
4521 if((cr36 & 0x0f) == 0x03) {
4522 r30 = 0x20; r31 = 0x12; r32 = 0xd0; r33 = 0x6b;
4523 }
4524 }
4525#endif
4526#ifdef ASUS_HACK
4527 if(SiS_Pr->SiS_CustomT == CUT_ASUSA2H_2) {
4528 if((cr36 & 0x0f) == 0x02) {
4529 /* r30 = 0x60; r31 = 0x05; r32 = 0x60; r33 = 0x33; */ /* rev 2 */
4530 /* r30 = 0x20; r31 = 0x05; r32 = 0x60; r33 = 0x33; */ /* rev 3 */
4531 /* r30 = 0x60; r31 = 0x0d; r32 = 0x70; r33 = 0x40; */ /* rev 4 */
4532 /* r30 = 0x20; r31 = 0x0d; r32 = 0x70; r33 = 0x40; */ /* rev 5 */
4533 }
4534 }
4535#endif
4536 }
4537
4538 if(!(SiS_Pr->OverruleEMI && (!r30) && (!r31) && (!r32) && (!r33))) {
4539 SiS_SetRegOR(SiS_Pr->SiS_Part4Port,0x30,0x20); /* Reset */
4540 SiS_GenericDelay(SiS_Pr, 0x500);
4541 }
4542 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x31,r31);
4543 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x32,r32);
4544 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x33,r33);
4545#endif /* SET_EMI */
4546
4547 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x34,0x10);
4548
4549#ifdef SET_EMI
4550 if( (SiS_LCDAEnabled(SiS_Pr, HwInfo)) ||
4551 (SiS_CRT2IsLCD(SiS_Pr, HwInfo)) ) {
4552 if(r30 & 0x40) {
4553 SiS_PanelDelayLoop(SiS_Pr, HwInfo, 3, 5);
4554 if(delaylong) {
4555 SiS_PanelDelayLoop(SiS_Pr, HwInfo, 3, 5);
4556 delaylong = FALSE;
4557 }
4558 SiS_WaitVBRetrace(SiS_Pr,HwInfo);
4559 if(SiS_Pr->SiS_CustomT == CUT_ASUSA2H_2) {
4560 SiS_GenericDelay(SiS_Pr, 0x500);
4561 }
4562 SiS_SetRegOR(SiS_Pr->SiS_Part4Port,0x30,0x40); /* Enable */
4563 }
4564 }
4565#endif
4566 }
4567 }
4568
4569 if(!(SiS_WeHaveBacklightCtrl(SiS_Pr,HwInfo))) {
4570 if(SiS_IsVAorLCD(SiS_Pr, HwInfo)) {
4571 SiS_PanelDelayLoop(SiS_Pr, HwInfo, 3, 10);
4572 if(delaylong) {
4573 SiS_PanelDelayLoop(SiS_Pr, HwInfo, 3, 10);
4574 }
4575 SiS_WaitVBRetrace(SiS_Pr,HwInfo);
4576 if(SiS_Pr->SiS_VBType & (VB_SIS302LV | VB_SIS302ELV)) {
4577 SiS_GenericDelay(SiS_Pr, 0x500);
4578 }
4579 SiS_SetRegOR(SiS_Pr->SiS_Part4Port,0x26,0x01);
4580 }
4581 }
4582
4583 SiS_SetReg(SiS_Pr->SiS_P3c4,0x06,pushax);
4584 SiS_DisplayOn(SiS_Pr);
4585 SiS_SetRegByte(SiS_Pr->SiS_P3c6,0xff);
4586
4587 }
4588
4589 if(!(SiS_WeHaveBacklightCtrl(SiS_Pr,HwInfo))) {
4590 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x00,0x7f);
4591 }
4592
4593#endif /* SIS315H */
4594
4595 }
4596
4597 } else { /* ============ For 301 ================ */
4598
4599 if(HwInfo->jChipType < SIS_315H) {
4600 if(SiS_CRT2IsLCD(SiS_Pr, HwInfo)) {
4601 SiS_SetRegSR11ANDOR(SiS_Pr,HwInfo,0xFB,0x00);
4602 SiS_PanelDelay(SiS_Pr, HwInfo, 0);
4603 }
4604 }
4605
4606 temp = SiS_GetReg(SiS_Pr->SiS_P3c4,0x32) & 0xDF; /* lock mode */
4607 if(SiS_BridgeInSlavemode(SiS_Pr)) {
4608 tempah = SiS_GetReg(SiS_Pr->SiS_P3d4,0x30);
4609 if(!(tempah & SetCRT2ToRAMDAC)) temp |= 0x20;
4610 }
4611 SiS_SetReg(SiS_Pr->SiS_P3c4,0x32,temp);
4612
4613 SiS_SetRegOR(SiS_Pr->SiS_P3c4,0x1E,0x20); /* enable CRT2 */
4614
4615 if(HwInfo->jChipType >= SIS_315H) {
4616 temp = SiS_GetReg(SiS_Pr->SiS_Part1Port,0x2E);
4617 if(!(temp & 0x80)) {
4618 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x2E,0x80); /* BVBDOENABLE=1 */
4619 }
4620 }
4621
4622 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x00,0x1F,0x20); /* enable VB processor */
4623
4624 SiS_VBLongWait(SiS_Pr);
4625 SiS_DisplayOn(SiS_Pr);
4626 if(HwInfo->jChipType >= SIS_315H) {
4627 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x00,0x7f);
4628 }
4629 SiS_VBLongWait(SiS_Pr);
4630
4631 if(HwInfo->jChipType < SIS_315H) {
4632 if(SiS_CRT2IsLCD(SiS_Pr, HwInfo)) {
4633 SiS_PanelDelay(SiS_Pr, HwInfo, 1);
4634 SiS_SetRegSR11ANDOR(SiS_Pr,HwInfo,0xF7,0x00);
4635 }
4636 }
4637
4638 }
4639
4640 } else { /* =================== For LVDS ================== */
4641
4642 if(HwInfo->jChipType < SIS_315H) {
4643
4644#ifdef SIS300 /* 300 series */
4645
4646 if(SiS_CRT2IsLCD(SiS_Pr, HwInfo)) {
4647 if(HwInfo->jChipType == SIS_730) {
4648 SiS_PanelDelay(SiS_Pr, HwInfo, 1);
4649 SiS_PanelDelay(SiS_Pr, HwInfo, 1);
4650 SiS_PanelDelay(SiS_Pr, HwInfo, 1);
4651 }
4652 SiS_SetRegSR11ANDOR(SiS_Pr,HwInfo,0xFB,0x00);
4653 if(!(SiS_CR36BIOSWord23d(SiS_Pr,HwInfo))) {
4654 SiS_PanelDelay(SiS_Pr, HwInfo, 0);
4655 }
4656 }
4657
4658 SiS_SetRegOR(SiS_Pr->SiS_P3c4,0x1E,0x20);
4659 SiS_DisplayOn(SiS_Pr);
4660 SiS_UnLockCRT2(SiS_Pr,HwInfo);
4661 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x02,0xBF);
4662 if(SiS_BridgeInSlavemode(SiS_Pr)) {
4663 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x01,0x1F);
4664 } else {
4665 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x01,0x1F,0x40);
4666 }
4667
4668 if(SiS_Pr->SiS_IF_DEF_CH70xx == 1) {
4669 if(!(SiS_CRT2IsLCD(SiS_Pr, HwInfo))) {
4670 SiS_WaitVBRetrace(SiS_Pr, HwInfo);
4671 SiS_SetCH700x(SiS_Pr,0x0B0E);
4672 }
4673 }
4674
4675 if(SiS_CRT2IsLCD(SiS_Pr, HwInfo)) {
4676 if(!(SiS_GetReg(SiS_Pr->SiS_P3c4,0x13) & 0x40)) {
4677 if(!(SiS_GetReg(SiS_Pr->SiS_P3c4,0x16) & 0x10)) {
4678 if(!(SiS_CR36BIOSWord23b(SiS_Pr,HwInfo))) {
4679 SiS_PanelDelay(SiS_Pr, HwInfo, 1);
4680 SiS_PanelDelay(SiS_Pr, HwInfo, 1);
4681 }
4682 SiS_WaitVBRetrace(SiS_Pr, HwInfo);
4683 SiS_SetRegSR11ANDOR(SiS_Pr,HwInfo,0xF7,0x00);
4684 }
4685 }
4686 }
4687
4688#endif /* SIS300 */
4689
4690 } else {
4691
4692#ifdef SIS315H /* 315 series */
4693
4694 if(!(SiS_IsNotM650orLater(SiS_Pr,HwInfo))) {
4695 if(HwInfo->jChipType < SIS_340) {
4696 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x4c,0x18);
4697 }
4698 }
4699
4700 if(SiS_Pr->SiS_IF_DEF_CH70xx == 0) {
4701 if(SiS_CRT2IsLCD(SiS_Pr, HwInfo)) {
4702 SiS_SetRegSR11ANDOR(SiS_Pr,HwInfo,0xFB,0x00);
4703 SiS_PanelDelay(SiS_Pr, HwInfo, 0);
4704 }
4705 }
4706
4707 SiS_SetRegOR(SiS_Pr->SiS_P3c4,0x1E,0x20);
4708 SiS_UnLockCRT2(SiS_Pr,HwInfo);
4709
4710 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x2e,0xf7);
4711
4712 if(SiS_Pr->SiS_IF_DEF_CH70xx == 2) {
4713 temp = SiS_GetCH701x(SiS_Pr,0x66);
4714 temp &= 0x20;
4715 SiS_Chrontel701xBLOff(SiS_Pr);
4716 }
4717
4718 if(HwInfo->jChipType != SIS_550) {
4719 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x2e,0x7f);
4720 }
4721
4722 if(HwInfo->jChipType == SIS_740) {
4723 if(SiS_Pr->SiS_IF_DEF_CH70xx == 2) {
4724 if(SiS_IsLCDOrLCDA(SiS_Pr, HwInfo)) {
4725 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x1E,0x20);
4726 }
4727 }
4728 }
4729
4730 temp1 = SiS_GetReg(SiS_Pr->SiS_Part1Port,0x2E);
4731 if(!(temp1 & 0x80)) {
4732 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x2E,0x80);
4733 }
4734
4735 if(SiS_Pr->SiS_IF_DEF_CH70xx == 2) {
4736 if(temp) {
4737 SiS_Chrontel701xBLOn(SiS_Pr, HwInfo);
4738 }
4739 }
4740
4741 if(SiS_Pr->SiS_IF_DEF_CH70xx == 0) {
4742 if(SiS_CRT2IsLCD(SiS_Pr, HwInfo)) {
4743 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x1E,0x20);
4744 if(HwInfo->jChipType == SIS_550) {
4745 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x1E,0x40);
4746 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x1E,0x10);
4747 }
4748 }
4749 } else if(SiS_IsVAMode(SiS_Pr,HwInfo)) {
4750 if(HwInfo->jChipType != SIS_740) {
4751 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x1E,0x20);
4752 }
4753 }
4754
4755 if(!(SiS_WeHaveBacklightCtrl(SiS_Pr,HwInfo))) {
4756 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x00,0x7f);
4757 }
4758
4759 if(SiS_Pr->SiS_IF_DEF_CH70xx == 2) {
4760 if(SiS_IsTVOrYPbPrOrScart(SiS_Pr,HwInfo)) {
4761 SiS_Chrontel701xOn(SiS_Pr,HwInfo);
4762 }
4763 if( (SiS_IsVAMode(SiS_Pr,HwInfo)) ||
4764 (SiS_IsLCDOrLCDA(SiS_Pr,HwInfo)) ) {
4765 SiS_ChrontelDoSomething1(SiS_Pr,HwInfo);
4766 }
4767 }
4768
4769 if(SiS_Pr->SiS_IF_DEF_CH70xx == 2) {
4770 if(!(SiS_WeHaveBacklightCtrl(SiS_Pr,HwInfo))) {
4771 if( (SiS_IsVAMode(SiS_Pr,HwInfo)) ||
4772 (SiS_IsLCDOrLCDA(SiS_Pr,HwInfo)) ) {
4773 SiS_Chrontel701xBLOn(SiS_Pr, HwInfo);
4774 SiS_ChrontelInitTVVSync(SiS_Pr,HwInfo);
4775 }
4776 }
4777 } else if(SiS_Pr->SiS_IF_DEF_CH70xx == 0) {
4778 if(!(SiS_WeHaveBacklightCtrl(SiS_Pr,HwInfo))) {
4779 if(SiS_CRT2IsLCD(SiS_Pr, HwInfo)) {
4780 SiS_PanelDelay(SiS_Pr, HwInfo, 1);
4781 SiS_SetRegSR11ANDOR(SiS_Pr,HwInfo,0xF7,0x00);
4782 }
4783 }
4784 }
4785
4786#endif /* SIS315H */
4787
4788 } /* 310 series */
4789
4790 } /* LVDS */
4791
4792}
4793
4794/*********************************************/
4795/* SET PART 1 REGISTER GROUP */
4796/*********************************************/
4797
4798/* Set CRT2 OFFSET / PITCH */
4799static void
4800SiS_SetCRT2Offset(SiS_Private *SiS_Pr, USHORT ModeNo, USHORT ModeIdIndex,
4801 USHORT RRTI, PSIS_HW_INFO HwInfo)
4802{
4803 USHORT offset;
4804 UCHAR temp;
4805
4806 if(SiS_Pr->SiS_VBInfo & SetInSlaveMode) return;
4807
4808 offset = SiS_GetOffset(SiS_Pr,ModeNo,ModeIdIndex,RRTI,HwInfo);
4809
4810 if((SiS_Pr->SiS_LCDResInfo == Panel_640x480_2) ||
4811 (SiS_Pr->SiS_LCDResInfo == Panel_640x480_3)) {
4812 offset >>= 1;
4813 }
4814
4815 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x07,(offset & 0xFF));
4816 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x09,(offset >> 8));
4817 temp = (UCHAR)(((offset >> 3) & 0xFF) + 1);
4818 if(offset % 8) temp++;
4819 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x03,temp);
4820}
4821
4822/* Set CRT2 sync and PanelLink mode */
4823static void
4824SiS_SetCRT2Sync(SiS_Private *SiS_Pr, USHORT ModeNo, USHORT RefreshRateTableIndex,
4825 PSIS_HW_INFO HwInfo)
4826{
4827 USHORT tempah=0,tempbl,infoflag;
4828
4829 tempbl = 0xC0;
4830
4831 if(SiS_Pr->UseCustomMode) {
4832 infoflag = SiS_Pr->CInfoFlag;
4833 } else {
4834 infoflag = SiS_Pr->SiS_RefIndex[RefreshRateTableIndex].Ext_InfoFlag;
4835 }
4836
4837 if(SiS_Pr->SiS_IF_DEF_LVDS == 1) { /* LVDS */
4838
4839 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) {
4840 tempah = 0;
4841 } else if((SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) && (SiS_Pr->SiS_LCDInfo & LCDSync)) {
4842 tempah = SiS_Pr->SiS_LCDInfo;
4843 } else tempah = infoflag >> 8;
4844 tempah &= 0xC0;
4845 tempah |= 0x20;
4846 if(!(SiS_Pr->SiS_LCDInfo & LCDRGB18Bit)) tempah |= 0x10;
4847 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) {
4848 if((SiS_Pr->SiS_CustomT == CUT_BARCO1366) ||
4849 (SiS_Pr->SiS_CustomT == CUT_BARCO1024)) {
4850 tempah |= 0xf0;
4851 }
4852 if( (SiS_Pr->SiS_IF_DEF_FSTN) ||
4853 (SiS_Pr->SiS_IF_DEF_DSTN) ||
4854 (SiS_Pr->SiS_IF_DEF_TRUMPION) ||
4855 (SiS_Pr->SiS_CustomT == CUT_PANEL848) ) {
4856 tempah |= 0x30;
4857 }
4858 }
4859 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) {
4860 if(HwInfo->jChipType >= SIS_315H) {
4861 tempah >>= 3;
4862 tempah &= 0x18;
4863 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x13,0xE7,tempah);
4864 /* Don't care about 12/18/24 bit mode - TV is via VGA, not PL */
4865 } else {
4866 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x19,0x0F,0xe0);
4867 }
4868 } else {
4869 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x19,0x0F,tempah);
4870 }
4871
4872 } else if(SiS_Pr->SiS_VBType & VB_SISVB) {
4873
4874 if(HwInfo->jChipType < SIS_315H) {
4875
4876#ifdef SIS300 /* ---- 300 series --- */
4877
4878 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) { /* 630 - 301B(-DH) */
4879
4880 tempah = infoflag >> 8;
4881 tempbl = 0;
4882 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) {
4883 if(SiS_Pr->SiS_LCDInfo & LCDSync) {
4884 tempah = SiS_Pr->SiS_LCDInfo;
4885 tempbl = (tempah >> 6) & 0x03;
4886 }
4887 }
4888 tempah &= 0xC0;
4889 tempah |= 0x20;
4890 if(!(SiS_Pr->SiS_LCDInfo & LCDRGB18Bit)) tempah |= 0x10;
4891 tempah |= 0xc0;
4892 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x19,0x0F,tempah);
4893 if((SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) && (!(SiS_Pr->SiS_VBType & VB_NoLCD))) {
4894 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x1a,0xf0,tempbl);
4895 }
4896
4897 } else { /* 630 - 301 */
4898
4899 tempah = infoflag >> 8;
4900 tempah &= 0xC0;
4901 tempah |= 0x20;
4902 if(!(SiS_Pr->SiS_LCDInfo & LCDRGB18Bit)) tempah |= 0x10;
4903 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x19,0x0F,tempah);
4904
4905 }
4906
4907#endif /* SIS300 */
4908
4909 } else {
4910
4911#ifdef SIS315H /* ------- 315 series ------ */
4912
4913 if(SiS_Pr->SiS_VBType & VB_SIS301LV302LV) { /* 315 - LVDS */
4914
4915 tempbl = 0;
4916 if((SiS_Pr->SiS_CustomT == CUT_COMPAQ1280) &&
4917 (SiS_Pr->SiS_LCDResInfo == Panel_1280x1024)) {
4918 tempah = infoflag >> 8;
4919 if(SiS_Pr->SiS_LCDInfo & LCDSync) {
4920 tempbl = ((SiS_Pr->SiS_LCDInfo & 0xc0) >> 6);
4921 }
4922 } else if((SiS_Pr->SiS_CustomT == CUT_CLEVO1400) &&
4923 (SiS_Pr->SiS_LCDResInfo == Panel_1400x1050)) {
4924 tempah = infoflag >> 8;
4925 tempbl = 0x03;
4926 } else {
4927 tempah = SiS_GetReg(SiS_Pr->SiS_P3d4,0x37);
4928 tempbl = (tempah >> 6) & 0x03;
4929 tempbl |= 0x08;
4930 if(!(SiS_Pr->SiS_LCDInfo & LCDRGB18Bit)) tempbl |= 0x04;
4931 }
4932 tempah &= 0xC0;
4933 tempah |= 0x20;
4934 if(!(SiS_Pr->SiS_LCDInfo & LCDRGB18Bit)) tempah |= 0x10;
4935 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA) tempah |= 0xc0;
4936 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x19,0x0F,tempah);
4937 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) {
4938 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) {
4939 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x1a,0xf0,tempbl);
4940 }
4941 }
4942
4943 } else { /* 315 - TMDS */
4944
4945 tempah = tempbl = infoflag >> 8;
4946 if(!SiS_Pr->UseCustomMode) {
4947 tempbl = 0;
4948 if((SiS_Pr->SiS_VBType & VB_SIS301C) && (SiS_Pr->SiS_VBInfo & SetCRT2ToRAMDAC)) {
4949 if(ModeNo <= 0x13) {
4950 tempah = SiS_GetRegByte((SiS_Pr->SiS_P3ca+0x02));
4951 }
4952 }
4953 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA)) {
4954 if(!(SiS_Pr->SiS_LCDInfo & LCDPass11)) {
4955 if(SiS_Pr->SiS_LCDInfo & LCDSync) {
4956 tempah = SiS_Pr->SiS_LCDInfo;
4957 tempbl = (tempah >> 6) & 0x03;
4958 }
4959 }
4960 }
4961 }
4962 tempah &= 0xC0;
4963 tempah |= 0x20;
4964 if(!(SiS_Pr->SiS_LCDInfo & LCDRGB18Bit)) tempah |= 0x10;
4965 if(SiS_Pr->SiS_VBType & VB_NoLCD) {
4966 /* Imitate BIOS bug */
4967 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) tempah |= 0xc0;
4968 }
4969 if((SiS_Pr->SiS_VBType & VB_SIS301C) && (SiS_Pr->SiS_VBInfo & SetCRT2ToRAMDAC)) {
4970 tempah >>= 3;
4971 tempah &= 0x18;
4972 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x13,0xe7,tempah);
4973 } else {
4974 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x19,0x0F,tempah);
4975 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) {
4976 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) {
4977 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x1a,0xf0,tempbl);
4978 }
4979 }
4980 }
4981
4982 }
4983#endif /* SIS315H */
4984 }
4985 }
4986}
4987
4988/* Set CRT2 FIFO on 300/630/730 */
4989#ifdef SIS300
4990static void
4991SiS_SetCRT2FIFO_300(SiS_Private *SiS_Pr,USHORT ModeNo,
4992 PSIS_HW_INFO HwInfo)
4993{
4994 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
4995 USHORT temp,index;
4996 USHORT modeidindex,refreshratetableindex;
4997 USHORT VCLK=0,MCLK,colorth=0,data2=0;
4998 USHORT tempal, tempah, tempbx, tempcl, tempax;
4999 USHORT CRT1ModeNo,CRT2ModeNo;
5000 USHORT SelectRate_backup;
5001 ULONG data,eax;
5002 const UCHAR LatencyFactor[] = {
5003 97, 88, 86, 79, 77, 00, /*; 64 bit BQ=2 */
5004 00, 87, 85, 78, 76, 54, /*; 64 bit BQ=1 */
5005 97, 88, 86, 79, 77, 00, /*; 128 bit BQ=2 */
5006 00, 79, 77, 70, 68, 48, /*; 128 bit BQ=1 */
5007 80, 72, 69, 63, 61, 00, /*; 64 bit BQ=2 */
5008 00, 70, 68, 61, 59, 37, /*; 64 bit BQ=1 */
5009 86, 77, 75, 68, 66, 00, /*; 128 bit BQ=2 */
5010 00, 68, 66, 59, 57, 37 /*; 128 bit BQ=1 */
5011 };
5012 const UCHAR LatencyFactor730[] = {
5013 69, 63, 61,
5014 86, 79, 77,
5015 103, 96, 94,
5016 120,113,111,
5017 137,130,128, /* <-- last entry, data below */
5018 137,130,128, /* to avoid using illegal values */
5019 137,130,128,
5020 137,130,128,
5021 137,130,128,
5022 137,130,128,
5023 137,130,128,
5024 137,130,128,
5025 137,130,128,
5026 137,130,128,
5027 137,130,128,
5028 137,130,128,
5029 };
5030 const UCHAR ThLowB[] = {
5031 81, 4, 72, 6, 88, 8,120,12,
5032 55, 4, 54, 6, 66, 8, 90,12,
5033 42, 4, 45, 6, 55, 8, 75,12
5034 };
5035 const UCHAR ThTiming[] = {
5036 1, 2, 2, 3, 0, 1, 1, 2
5037 };
5038
5039 SelectRate_backup = SiS_Pr->SiS_SelectCRT2Rate;
5040
5041 if(!SiS_Pr->CRT1UsesCustomMode) {
5042
5043 CRT1ModeNo = SiS_Pr->SiS_CRT1Mode; /* get CRT1 ModeNo */
5044 SiS_SearchModeID(SiS_Pr, &CRT1ModeNo, &modeidindex);
5045 SiS_Pr->SiS_SetFlag &= (~ProgrammingCRT2);
5046 SiS_Pr->SiS_SelectCRT2Rate = 0;
5047 refreshratetableindex = SiS_GetRatePtr(SiS_Pr, CRT1ModeNo, modeidindex, HwInfo);
5048
5049 if(CRT1ModeNo >= 0x13) {
5050 index = SiS_Pr->SiS_RefIndex[refreshratetableindex].Ext_CRTVCLK;
5051 index &= 0x3F;
5052 VCLK = SiS_Pr->SiS_VCLKData[index].CLOCK; /* Get VCLK */
5053
5054 colorth = SiS_GetColorDepth(SiS_Pr,CRT1ModeNo,modeidindex); /* Get colordepth */
5055 colorth >>= 1;
5056 if(!colorth) colorth++;
5057 }
5058
5059 } else {
5060
5061 CRT1ModeNo = 0xfe;
5062 VCLK = SiS_Pr->CSRClock_CRT1; /* Get VCLK */
5063 data2 = (SiS_Pr->CModeFlag_CRT1 & ModeTypeMask) - 2;
5064 switch(data2) { /* Get color depth */
5065 case 0 : colorth = 1; break;
5066 case 1 : colorth = 1; break;
5067 case 2 : colorth = 2; break;
5068 case 3 : colorth = 2; break;
5069 case 4 : colorth = 3; break;
5070 case 5 : colorth = 4; break;
5071 default: colorth = 2;
5072 }
5073
5074 }
5075
5076 if(CRT1ModeNo >= 0x13) {
5077 if(HwInfo->jChipType == SIS_300) {
5078 index = SiS_GetReg(SiS_Pr->SiS_P3c4,0x3A);
5079 } else {
5080 index = SiS_GetReg(SiS_Pr->SiS_P3c4,0x1A);
5081 }
5082 index &= 0x07;
5083 MCLK = SiS_Pr->SiS_MCLKData_0[index].CLOCK; /* Get MCLK */
5084
5085 data2 = (colorth * VCLK) / MCLK;
5086
5087 temp = SiS_GetReg(SiS_Pr->SiS_P3c4,0x14);
5088 temp = ((temp & 0x00FF) >> 6) << 1;
5089 if(temp == 0) temp = 1;
5090 temp <<= 2;
5091 temp &= 0xff;
5092
5093 data2 = temp - data2;
5094
5095 if((28 * 16) % data2) {
5096 data2 = (28 * 16) / data2;
5097 data2++;
5098 } else {
5099 data2 = (28 * 16) / data2;
5100 }
5101
5102 if(HwInfo->jChipType == SIS_300) {
5103
5104 tempah = SiS_GetReg(SiS_Pr->SiS_P3c4,0x18);
5105 tempah &= 0x62;
5106 tempah >>= 1;
5107 tempal = tempah;
5108 tempah >>= 3;
5109 tempal |= tempah;
5110 tempal &= 0x07;
5111 tempcl = ThTiming[tempal];
5112 tempbx = SiS_GetReg(SiS_Pr->SiS_P3c4,0x16);
5113 tempbx >>= 6;
5114 tempah = SiS_GetReg(SiS_Pr->SiS_P3c4,0x14);
5115 tempah >>= 4;
5116 tempah &= 0x0c;
5117 tempbx |= tempah;
5118 tempbx <<= 1;
5119 tempal = ThLowB[tempbx + 1];
5120 tempal *= tempcl;
5121 tempal += ThLowB[tempbx];
5122 data = tempal;
5123
5124 } else if(HwInfo->jChipType == SIS_730) {
5125
5126#ifdef LINUX_KERNEL
5127 SiS_SetRegLong(0xcf8,0x80000050);
5128 eax = SiS_GetRegLong(0xcfc);
5129#else
5130 eax = pciReadLong(0x00000000, 0x50);
5131#endif
5132 tempal = (USHORT)(eax >> 8);
5133 tempal &= 0x06;
5134 tempal <<= 5;
5135
5136#ifdef LINUX_KERNEL
5137 SiS_SetRegLong(0xcf8,0x800000A0);
5138 eax = SiS_GetRegLong(0xcfc);
5139#else
5140 eax = pciReadLong(0x00000000, 0xA0);
5141#endif
5142 temp = (USHORT)(eax >> 28);
5143 temp &= 0x0F;
5144 tempal |= temp;
5145
5146 tempbx = tempal; /* BIOS BUG (2.04.5d, 2.04.6a use ah here, which is unset!) */
5147 tempbx = 0; /* -- do it like the BIOS anyway... */
5148 tempax = tempbx;
5149 tempbx &= 0xc0;
5150 tempbx >>= 6;
5151 tempax &= 0x0f;
5152 tempax *= 3;
5153 tempbx += tempax;
5154
5155 data = LatencyFactor730[tempbx];
5156 data += 15;
5157 temp = SiS_GetReg(SiS_Pr->SiS_P3c4,0x14);
5158 if(!(temp & 0x80)) data += 5;
5159
5160 } else {
5161
5162 index = 0;
5163 temp = SiS_GetReg(SiS_Pr->SiS_P3c4,0x14);
5164 if(temp & 0x0080) index += 12;
5165
5166#ifdef LINUX_KERNEL
5167 SiS_SetRegLong(0xcf8,0x800000A0);
5168 eax = SiS_GetRegLong(0xcfc);
5169#else
5170 /* We use pci functions X offers. We use tag 0, because
5171 * we want to read/write to the host bridge (which is always
5172 * 00:00.0 on 630, 730 and 540), not the VGA device.
5173 */
5174 eax = pciReadLong(0x00000000, 0xA0);
5175#endif
5176 temp = (USHORT)(eax >> 24);
5177 if(!(temp&0x01)) index += 24;
5178
5179#ifdef LINUX_KERNEL
5180 SiS_SetRegLong(0xcf8,0x80000050);
5181 eax = SiS_GetRegLong(0xcfc);
5182#else
5183 eax = pciReadLong(0x00000000, 0x50);
5184#endif
5185 temp=(USHORT)(eax >> 24);
5186 if(temp & 0x01) index += 6;
5187
5188 temp = (temp & 0x0F) >> 1;
5189 index += temp;
5190
5191 data = LatencyFactor[index];
5192 data += 15;
5193 temp = SiS_GetReg(SiS_Pr->SiS_P3c4,0x14);
5194 if(!(temp & 0x80)) data += 5;
5195 }
5196
5197 data += data2; /* CRT1 Request Period */
5198
5199 SiS_Pr->SiS_SetFlag |= ProgrammingCRT2;
5200 SiS_Pr->SiS_SelectCRT2Rate = SelectRate_backup;
5201
5202 if(!SiS_Pr->UseCustomMode) {
5203
5204 CRT2ModeNo = ModeNo;
5205 SiS_SearchModeID(SiS_Pr, &CRT2ModeNo, &modeidindex);
5206
5207 refreshratetableindex = SiS_GetRatePtr(SiS_Pr, CRT2ModeNo, modeidindex, HwInfo);
5208
5209 index = SiS_GetVCLK2Ptr(SiS_Pr,CRT2ModeNo,modeidindex,
5210 refreshratetableindex,HwInfo);
5211 VCLK = SiS_Pr->SiS_VCLKData[index].CLOCK; /* Get VCLK */
5212
5213 if((SiS_Pr->SiS_CustomT == CUT_BARCO1366) || (SiS_Pr->SiS_CustomT == CUT_BARCO1024)) {
5214 if(SiS_Pr->SiS_UseROM) {
5215 if(ROMAddr[0x220] & 0x01) {
5216 VCLK = ROMAddr[0x229] | (ROMAddr[0x22a] << 8);
5217 }
5218 }
5219 }
5220
5221 } else {
5222
5223 CRT2ModeNo = 0xfe;
5224 VCLK = SiS_Pr->CSRClock; /* Get VCLK */
5225
5226 }
5227
5228 colorth = SiS_GetColorDepth(SiS_Pr,CRT2ModeNo,modeidindex); /* Get colordepth */
5229 colorth >>= 1;
5230 if(!colorth) colorth++;
5231
5232 data = data * VCLK * colorth;
5233 if(data % (MCLK << 4)) {
5234 data = data / (MCLK << 4);
5235 data++;
5236 } else {
5237 data = data / (MCLK << 4);
5238 }
5239
5240 if(data <= 6) data = 6;
5241 if(data > 0x14) data = 0x14;
5242
5243 temp = SiS_GetReg(SiS_Pr->SiS_Part1Port,0x01);
5244 if(HwInfo->jChipType == SIS_300) {
5245 if(data <= 0x0f) temp = (temp & (~0x1F)) | 0x13;
5246 else temp = (temp & (~0x1F)) | 0x16;
5247 if(SiS_Pr->SiS_LCDResInfo == Panel_1280x1024) {
5248 temp = (temp & (~0x1F)) | 0x13;
5249 }
5250 } else {
5251 if( ( (HwInfo->jChipType == SIS_630) ||
5252 (HwInfo->jChipType == SIS_730) ) &&
5253 (HwInfo->jChipRevision >= 0x30) ) /* 630s or 730(s?) */
5254 {
5255 temp = (temp & (~0x1F)) | 0x1b;
5256 } else {
5257 temp = (temp & (~0x1F)) | 0x16;
5258 }
5259 }
5260 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x01,0xe0,temp);
5261
5262 if( (HwInfo->jChipType == SIS_630) &&
5263 (HwInfo->jChipRevision >= 0x30) ) /* 630s, NOT 730 */
5264 {
5265 if(data > 0x13) data = 0x13;
5266 }
5267 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x02,0xe0,data);
5268
5269 } else { /* If mode <= 0x13, we just restore everything */
5270
5271 SiS_Pr->SiS_SetFlag |= ProgrammingCRT2;
5272 SiS_Pr->SiS_SelectCRT2Rate = SelectRate_backup;
5273
5274 }
5275}
5276#endif
5277
5278/* Set CRT2 FIFO on 315/330 series */
5279#ifdef SIS315H
5280static void
5281SiS_SetCRT2FIFO_310(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
5282{
5283 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x01,0x3B);
5284 if( (HwInfo->jChipType == SIS_760) &&
5285 (SiS_Pr->SiS_SysFlags & SF_760LFB) &&
5286 (SiS_Pr->SiS_ModeType == Mode32Bpp) &&
5287 (SiS_Pr->SiS_VGAHDE >= 1280) &&
5288 (SiS_Pr->SiS_VGAVDE >= 1024) ) {
5289 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x2f,0x03);
5290 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x01,0x3b);
5291 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x4d,0xc0);
5292 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x2f,0x01);
5293 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x4d,0xc0);
5294 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x02,0x6e);
5295 } else {
5296 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x02,~0x3f,0x04);
5297 }
5298
5299}
5300#endif
5301
5302static USHORT
5303SiS_GetVGAHT2(SiS_Private *SiS_Pr)
5304{
5305 ULONG tempax,tempbx;
5306
5307 tempbx = (SiS_Pr->SiS_VGAVT - SiS_Pr->SiS_VGAVDE) * SiS_Pr->SiS_RVBHCMAX;
5308 tempax = (SiS_Pr->SiS_VT - SiS_Pr->SiS_VDE) * SiS_Pr->SiS_RVBHCFACT;
5309 tempax = (tempax * SiS_Pr->SiS_HT) / tempbx;
5310 return((USHORT)tempax);
5311}
5312
5313/* Set Part 1 / SiS bridge slave mode */
5314static void
5315SiS_SetGroup1_301(SiS_Private *SiS_Pr, USHORT ModeNo,USHORT ModeIdIndex,
5316 PSIS_HW_INFO HwInfo,USHORT RefreshRateTableIndex)
5317{
5318 USHORT push1,push2;
5319 USHORT tempax,tempbx,tempcx,temp;
5320 USHORT resinfo,modeflag,xres=0;
5321 unsigned char p1_7, p1_8;
5322
5323 if(ModeNo <= 0x13) {
5324 modeflag = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_ModeFlag;
5325 resinfo = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_ResInfo;
5326 } else if(SiS_Pr->UseCustomMode) {
5327 modeflag = SiS_Pr->CModeFlag;
5328 resinfo = 0;
5329 xres = SiS_Pr->CHDisplay;
5330 } else {
5331 modeflag = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_ModeFlag;
5332 resinfo = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_RESINFO;
5333 xres = SiS_Pr->SiS_RefIndex[RefreshRateTableIndex].XRes;
5334 }
5335
5336 /* The following is only done if bridge is in slave mode: */
5337
5338 if((HwInfo->jChipType >= SIS_661) && (ModeNo > 0x13)) {
5339 if(xres >= 1600) {
5340 SiS_SetRegOR(SiS_Pr->SiS_P3c4,0x31,0x04);
5341 }
5342 }
5343
5344 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x03,0xff); /* set MAX HT */
5345
5346 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) modeflag |= Charx8Dot;
5347
5348 if(modeflag & Charx8Dot) tempcx = 0x08;
5349 else tempcx = 0x09;
5350
5351 tempax = SiS_Pr->SiS_VGAHDE; /* 0x04 Horizontal Display End */
5352 if(modeflag & HalfDCLK) tempax >>= 1;
5353 tempax = ((tempax / tempcx) - 1) & 0xff;
5354 tempbx = tempax;
5355
5356 temp = tempax;
5357 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x04,temp);
5358
5359 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) {
5360 if(!(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV)) {
5361 temp += 2;
5362 }
5363 }
5364 if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) {
5365 if(resinfo == SIS_RI_800x600) temp -= 2;
5366 }
5367 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x05,temp); /* 0x05 Horizontal Display Start */
5368
5369 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x06,0x03); /* 0x06 Horizontal Blank end */
5370
5371 tempax = 0xFFFF;
5372 if(!(SiS_Pr->SiS_VBInfo & SetCRT2ToTV)) tempax = SiS_GetVGAHT2(SiS_Pr);
5373 if(tempax >= SiS_Pr->SiS_VGAHT) tempax = SiS_Pr->SiS_VGAHT;
5374 if(modeflag & HalfDCLK) tempax >>= 1;
5375 tempax = (tempax / tempcx) - 5;
5376 tempcx = tempax;
5377
5378 if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) {
5379 temp = tempcx - 1;
5380 if(!(modeflag & HalfDCLK)) {
5381 temp -= 6;
5382 if(SiS_Pr->SiS_TVMode & TVSetTVSimuMode) {
5383 temp -= 2;
5384 if(ModeNo > 0x13) temp -= 10;
5385 }
5386 }
5387 } else {
5388 tempcx = (tempcx + tempbx) >> 1;
5389 temp = (tempcx & 0x00FF) + 2;
5390 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) {
5391 temp--;
5392 if(!(modeflag & HalfDCLK)) {
5393 if((modeflag & Charx8Dot)) {
5394 temp += 4;
5395 if(SiS_Pr->SiS_VGAHDE >= 800) temp -= 6;
5396 if(HwInfo->jChipType >= SIS_315H) {
5397 if(SiS_Pr->SiS_VGAHDE == 800) temp += 2;
5398 }
5399 }
5400 }
5401 } else {
5402 if(!(modeflag & HalfDCLK)) {
5403 temp -= 4;
5404 if((SiS_Pr->SiS_LCDResInfo != Panel_1280x960) &&
5405 (SiS_Pr->SiS_LCDResInfo != Panel_1600x1200)) {
5406 if(SiS_Pr->SiS_VGAHDE >= 800) {
5407 temp -= 7;
5408 if(HwInfo->jChipType < SIS_315H) {
5409 if(SiS_Pr->SiS_ModeType == ModeEGA) {
5410 if(SiS_Pr->SiS_VGAVDE == 1024) {
5411 temp += 15;
5412 if(SiS_Pr->SiS_LCDResInfo != Panel_1280x1024)
5413 temp += 7;
5414 }
5415 }
5416 }
5417 if(SiS_Pr->SiS_LCDResInfo != Panel_1400x1050) {
5418 if(SiS_Pr->SiS_VGAHDE >= 1280) {
5419 if(SiS_Pr->SiS_LCDInfo & DontExpandLCD) temp += 28;
5420 }
5421 }
5422 }
5423 }
5424 }
5425 }
5426 }
5427
5428 p1_7 = temp;
5429 p1_8 = 0x00;
5430
5431 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) {
5432 if(SiS_Pr->SiS_TVMode & TVSetTVSimuMode) {
5433 if(ModeNo <= 0x01) {
5434 p1_7 = 0x2a;
5435 if(!(SiS_Pr->SiS_TVMode & TVSetPAL)) p1_8 = 0x61;
5436 else p1_8 = 0x41;
5437 } else if(SiS_Pr->SiS_ModeType == ModeText) {
5438 if(!(SiS_Pr->SiS_TVMode & TVSetPAL)) p1_7 = 0x54;
5439 else p1_7 = 0x55;
5440 p1_8 = 0x00;
5441 } else if(ModeNo <= 0x13) {
5442 if(modeflag & HalfDCLK) {
5443 if(!(SiS_Pr->SiS_TVMode & TVSetPAL)) {
5444 p1_7 = 0x30;
5445 p1_8 = 0x03;
5446 } else {
5447 p1_7 = 0x2f;
5448 p1_8 = 0x02;
5449 }
5450 } else {
5451 p1_7 = 0x5b;
5452 p1_8 = 0x03;
5453 }
5454 } else if( ((HwInfo->jChipType >= SIS_315H) &&
5455 ((ModeNo == 0x50) || (ModeNo == 0x56) || (ModeNo == 0x53))) ||
5456 ((HwInfo->jChipType < SIS_315H) &&
5457 (resinfo == SIS_RI_320x200 || resinfo == SIS_RI_320x240)) ) {
5458 if(!(SiS_Pr->SiS_TVMode & TVSetPAL)) {
5459 p1_7 = 0x30,
5460 p1_8 = 0x03;
5461 } else {
5462 p1_7 = 0x2f;
5463 p1_8 = 0x03;
5464 }
5465 }
5466 }
5467 }
5468
5469 if(SiS_Pr->SiS_VBInfo & SetCRT2ToYPbPr525750) {
5470 if(SiS_Pr->SiS_TVMode & (TVSetYPbPr525p|TVSetYPbPr750p)) {
5471 p1_7 = 0x63;
5472 if(SiS_Pr->SiS_TVMode & TVSetYPbPr750p) p1_7 = 0x55;
5473 }
5474 if(SiS_Pr->SiS_TVMode & TVSetTVSimuMode) {
5475 if(!(modeflag & HalfDCLK)) {
5476 p1_7 = 0xb2;
5477 if(SiS_Pr->SiS_TVMode & TVSetYPbPr750p) {
5478 p1_7 = 0xab;
5479 }
5480 }
5481 } else {
5482 if(SiS_Pr->SiS_TVMode & TVSetYPbPr525p) {
5483 if(modeflag & HalfDCLK) p1_7 = 0x30;
5484 }
5485 }
5486 }
5487
5488 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x07,p1_7); /* 0x07 Horizontal Retrace Start */
5489 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x08,p1_8); /* 0x08 Horizontal Retrace End */
5490
5491 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x18,0x03); /* 0x18 SR08 (FIFO Threshold?) */
5492
5493 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x19,0xF0);
5494
5495 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x09,0xFF); /* 0x09 Set Max VT */
5496
5497 tempcx = 0x121;
5498 tempbx = SiS_Pr->SiS_VGAVDE; /* 0x0E Vertical Display End */
5499 if (tempbx == 357) tempbx = 350;
5500 else if(tempbx == 360) tempbx = 350;
5501 else if(tempbx == 375) tempbx = 350;
5502 else if(tempbx == 405) tempbx = 400;
5503 else if(tempbx == 420) tempbx = 400;
5504 else if(tempbx == 525) tempbx = 480;
5505 push2 = tempbx;
5506 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) {
5507 if(SiS_Pr->SiS_LCDResInfo == Panel_1024x768) {
5508 if(!(SiS_Pr->SiS_SetFlag & LCDVESATiming)) {
5509 if (tempbx == 350) tempbx += 5;
5510 else if(tempbx == 480) tempbx += 5;
5511 }
5512 }
5513 }
5514 tempbx -= 2;
5515 temp = tempbx & 0x00FF;
5516 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x10,temp); /* 0x10 vertical Blank Start */
5517
5518 tempbx = push2;
5519 tempbx--;
5520 temp = tempbx & 0x00FF;
5521#if 0
5522 /* Missing code from 630/301B 2.04.5a and 650/302LV 1.10.6s (calles int 2f) */
5523 if(xxx()) {
5524 if(temp == 0xdf) temp = 0xda;
5525 }
5526#endif
5527 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x0E,temp);
5528
5529 temp = 0;
5530 if(modeflag & DoubleScanMode) temp |= 0x80;
5531 if(HwInfo->jChipType >= SIS_661) {
5532 if(tempbx & 0x0200) temp |= 0x20;
5533 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x0B,0x5F,temp);
5534 if(tempbx & 0x0100) tempcx |= 0x000a;
5535 if(tempbx & 0x0400) tempcx |= 0x1200;
5536 } else {
5537 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x0B,temp);
5538 if(tempbx & 0x0100) tempcx |= 0x0002;
5539 if(tempbx & 0x0400) tempcx |= 0x0600;
5540 }
5541
5542 if(tempbx & 0x0200) tempcx |= 0x0040;
5543
5544 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x11,0x00); /* 0x11 Vertical Blank End */
5545
5546 tempax = (SiS_Pr->SiS_VGAVT - tempbx) >> 2;
5547
5548 if((ModeNo > 0x13) || (HwInfo->jChipType < SIS_315H)) {
5549 if(resinfo != SIS_RI_1280x1024) {
5550 tempbx += (tempax << 1);
5551 }
5552 } else if(HwInfo->jChipType >= SIS_315H) {
5553 if(SiS_Pr->SiS_LCDResInfo != Panel_1400x1050) {
5554 tempbx += (tempax << 1);
5555 }
5556 }
5557
5558 if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) {
5559 tempbx -= 10;
5560 } else {
5561 if(SiS_Pr->SiS_TVMode & TVSetTVSimuMode) {
5562 if(SiS_Pr->SiS_TVMode & TVSetPAL) {
5563 tempbx += 40;
5564 if(HwInfo->jChipType >= SIS_315H) {
5565 if(SiS_Pr->SiS_VGAHDE == 800) tempbx += 10;
5566 }
5567 }
5568 }
5569 }
5570 tempax >>= 2;
5571 tempax++;
5572 tempax += tempbx;
5573 push1 = tempax;
5574 if(SiS_Pr->SiS_TVMode & TVSetPAL) {
5575 if(tempbx <= 513) {
5576 if(tempax >= 513) tempbx = 513;
5577 }
5578 }
5579 temp = tempbx & 0x00FF;
5580 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x0C,temp); /* 0x0C Vertical Retrace Start */
5581
5582 tempbx--;
5583 temp = tempbx & 0x00FF;
5584 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x10,temp);
5585
5586 if(tempbx & 0x0100) tempcx |= 0x0008;
5587
5588 if(tempbx & 0x0200) {
5589 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x0B,0x20);
5590 }
5591 tempbx++;
5592
5593 if(tempbx & 0x0100) tempcx |= 0x0004;
5594 if(tempbx & 0x0200) tempcx |= 0x0080;
5595 if(tempbx & 0x0400) {
5596 if(HwInfo->jChipType >= SIS_661) tempcx |= 0x0800;
5597 else if(SiS_Pr->SiS_VBType & VB_SIS301) tempcx |= 0x0800;
5598 else tempcx |= 0x0C00;
5599 }
5600
5601 tempbx = push1;
5602 temp = tempbx & 0x000F;
5603 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x0D,temp); /* 0x0D vertical Retrace End */
5604
5605 if(tempbx & 0x0010) tempcx |= 0x2000;
5606
5607 temp = tempcx & 0x00FF;
5608 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x0A,temp); /* 0x0A CR07 */
5609
5610 temp = (tempcx & 0xFF00) >> 8;
5611 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x17,temp); /* 0x17 SR0A */
5612
5613 tempax = modeflag;
5614 temp = (tempax & 0xFF00) >> 8;
5615 temp = (temp >> 1) & 0x09;
5616 if(!(SiS_Pr->SiS_VBType & VB_SIS301)) temp |= 0x01; /* Always 8 dotclock */
5617 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x16,temp); /* 0x16 SR01 */
5618
5619 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x0F,0x00); /* 0x0F CR14 */
5620
5621 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x12,0x00); /* 0x12 CR17 */
5622
5623 temp = 0x00;
5624 if(SiS_Pr->SiS_LCDInfo & LCDRGB18Bit) {
5625 if(SiS_GetReg(SiS_Pr->SiS_Part1Port,0x00) & 0x01) {
5626 temp = 0x80;
5627 }
5628 }
5629 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1A,temp); /* 0x1A SR0E */
5630
5631 temp = SiS_GetRegByte((SiS_Pr->SiS_P3ca+0x02));
5632 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1b,temp);
5633}
5634
5635/* Setup panel link
5636 * This is used for LVDS, LCDA and Chrontel TV output
5637 * 300/LVDS+TV, 300/301B-DH, 315/LVDS+TV, 315/LCDA
5638 */
5639static void
5640SiS_SetGroup1_LVDS(SiS_Private *SiS_Pr, USHORT ModeNo, USHORT ModeIdIndex,
5641 PSIS_HW_INFO HwInfo, USHORT RefreshRateTableIndex)
5642{
5643 USHORT modeflag,resinfo;
5644 USHORT push2,tempax,tempbx,tempcx,temp;
5645 ULONG tempeax=0,tempebx,tempecx,tempvcfact=0;
5646 BOOLEAN islvds = FALSE, issis = FALSE, chkdclkfirst = FALSE;
5647#ifdef SIS300
5648 USHORT crt2crtc;
5649#endif
5650#ifdef SIS315H
5651 USHORT pushcx;
5652#endif
5653
5654 if(ModeNo <= 0x13) {
5655 modeflag = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_ModeFlag;
5656 resinfo = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_ResInfo;
5657#ifdef SIS300
5658 crt2crtc = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_CRT2CRTC;
5659#endif
5660 } else if(SiS_Pr->UseCustomMode) {
5661 modeflag = SiS_Pr->CModeFlag;
5662 resinfo = 0;
5663#ifdef SIS300
5664 crt2crtc = 0;
5665#endif
5666 } else {
5667 modeflag = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_ModeFlag;
5668 resinfo = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_RESINFO;
5669#ifdef SIS300
5670 crt2crtc = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_CRT2CRTC;
5671#endif
5672 }
5673
5674 /* is lvds if really LVDS, or 301B-DH with external LVDS transmitter */
5675 if((SiS_Pr->SiS_IF_DEF_LVDS == 1) || (SiS_Pr->SiS_VBType & VB_NoLCD)) {
5676 islvds = TRUE;
5677 }
5678
5679 /* is really sis if sis bridge, but not 301B-DH */
5680 if((SiS_Pr->SiS_VBType & VB_SISVB) && (!(SiS_Pr->SiS_VBType & VB_NoLCD))) {
5681 issis = TRUE;
5682 }
5683
5684 if((HwInfo->jChipType >= SIS_315H) && (islvds) && (!(SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA))) {
5685 if((!SiS_Pr->SiS_IF_DEF_FSTN) && (!SiS_Pr->SiS_IF_DEF_DSTN)) {
5686 chkdclkfirst = TRUE;
5687 }
5688 }
5689
5690#ifdef SIS315H
5691 if((HwInfo->jChipType >= SIS_315H) && (SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA)) {
5692 if(IS_SIS330) {
5693 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x2D,0x10);
5694 } else if(IS_SIS740) {
5695 if(islvds) {
5696 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x13,0xfb,0x04);
5697 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x2D,0x03);
5698 } else if(SiS_Pr->SiS_VBType & VB_SISVB) {
5699 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x2D,0x10);
5700 }
5701 } else {
5702 if(islvds) {
5703 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x13,0xfb,0x04);
5704 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x2D,0x00);
5705 } else if(SiS_Pr->SiS_VBType & VB_SISVB) {
5706 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x2D,0x0f);
5707 if(SiS_Pr->SiS_VBType & VB_SIS301C) {
5708 if((SiS_Pr->SiS_LCDResInfo == Panel_1024x768) ||
5709 (SiS_Pr->SiS_LCDResInfo == Panel_1280x1024)) {
5710 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x2D,0x20);
5711 }
5712 }
5713 }
5714 }
5715 }
5716#endif
5717
5718 /* Horizontal */
5719
5720 tempax = SiS_Pr->SiS_LCDHDES;
5721 if(islvds) {
5722 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA)) {
5723 if((!SiS_Pr->SiS_IF_DEF_FSTN) && (!SiS_Pr->SiS_IF_DEF_DSTN)) {
5724 if((SiS_Pr->SiS_LCDResInfo == Panel_640x480) &&
5725 (!(SiS_Pr->SiS_VBInfo & SetInSlaveMode))) {
5726 tempax -= 8;
5727 }
5728 }
5729 }
5730 }
5731
5732 temp = (tempax & 0x0007);
5733 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1A,temp); /* BPLHDESKEW[2:0] */
5734 temp = (tempax >> 3) & 0x00FF;
5735 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x16,temp); /* BPLHDESKEW[10:3] */
5736
5737 tempbx = SiS_Pr->SiS_HDE;
5738 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA)) {
5739 if((SiS_Pr->SiS_LCDResInfo == Panel_640x480_2) ||
5740 (SiS_Pr->SiS_LCDResInfo == Panel_640x480_3)) {
5741 tempbx >>= 1;
5742 }
5743 if(!(SiS_Pr->SiS_LCDInfo & LCDPass11)) {
5744 tempbx = SiS_Pr->PanelXRes;
5745 }
5746 }
5747
5748 tempax += tempbx;
5749 if(tempax >= SiS_Pr->SiS_HT) tempax -= SiS_Pr->SiS_HT;
5750
5751 temp = tempax;
5752 if(temp & 0x07) temp += 8;
5753 temp >>= 3;
5754 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x17,temp); /* BPLHDEE */
5755
5756 tempcx = (SiS_Pr->SiS_HT - tempbx) >> 2;
5757
5758 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA)) {
5759 if(!(SiS_Pr->SiS_LCDInfo & LCDPass11)) {
5760 if(SiS_Pr->PanelHRS != 999) tempcx = SiS_Pr->PanelHRS;
5761 }
5762 }
5763
5764 tempcx += tempax;
5765 if(tempcx >= SiS_Pr->SiS_HT) tempcx -= SiS_Pr->SiS_HT;
5766
5767 temp = (tempcx >> 3) & 0x00FF;
5768 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA)) {
5769 if(SiS_Pr->SiS_IF_DEF_TRUMPION) {
5770 if(SiS_Pr->SiS_LCDResInfo == Panel_1024x768) {
5771 switch(ModeNo) {
5772 case 0x04:
5773 case 0x05:
5774 case 0x0d: temp = 0x56; break;
5775 case 0x10: temp = 0x60; break;
5776 case 0x13: temp = 0x5f; break;
5777 case 0x40:
5778 case 0x41:
5779 case 0x4f:
5780 case 0x43:
5781 case 0x44:
5782 case 0x62:
5783 case 0x56:
5784 case 0x53:
5785 case 0x5d:
5786 case 0x5e: temp = 0x54; break;
5787 }
5788 }
5789 }
5790 }
5791 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x14,temp); /* BPLHRS */
5792
5793 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA)) {
5794 temp += 2;
5795 if(!(SiS_Pr->SiS_LCDInfo & LCDPass11)) {
5796 temp += 8;
5797 if(SiS_Pr->PanelHRE != 999) {
5798 temp = tempcx + SiS_Pr->PanelHRE;
5799 if(temp >= SiS_Pr->SiS_HT) temp -= SiS_Pr->SiS_HT;
5800 temp >>= 3;
5801 }
5802 }
5803 } else {
5804 temp += 10;
5805 }
5806
5807 temp &= 0x1F;
5808 temp |= ((tempcx & 0x07) << 5);
5809#if 0
5810 if(SiS_Pr->SiS_IF_DEF_FSTN) temp = 0x20; /* WRONG? BIOS loads cl, not ah */
5811#endif
5812 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x15,temp); /* BPLHRE */
5813
5814 /* Vertical */
5815
5816 tempax = SiS_Pr->SiS_VGAVDE;
5817 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA)) {
5818 if(!(SiS_Pr->SiS_LCDInfo & LCDPass11)) {
5819 tempax = SiS_Pr->PanelYRes;
5820 }
5821 }
5822
5823 tempbx = SiS_Pr->SiS_LCDVDES + tempax;
5824 if(tempbx >= SiS_Pr->SiS_VT) tempbx -= SiS_Pr->SiS_VT;
5825
5826 push2 = tempbx;
5827
5828 tempcx = SiS_Pr->SiS_VGAVT - SiS_Pr->SiS_VGAVDE;
5829 if(HwInfo->jChipType < SIS_315H) {
5830 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA)) {
5831 if(!(SiS_Pr->SiS_LCDInfo & LCDPass11)) {
5832 tempcx = SiS_Pr->SiS_VGAVT - SiS_Pr->PanelYRes;
5833 }
5834 }
5835 }
5836 if(islvds) tempcx >>= 1;
5837 else tempcx >>= 2;
5838
5839 if( (SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA)) &&
5840 (!(SiS_Pr->SiS_LCDInfo & LCDPass11)) &&
5841 (SiS_Pr->PanelVRS != 999) ) {
5842 tempcx = SiS_Pr->PanelVRS;
5843 tempbx += tempcx;
5844 if(issis) tempbx++;
5845 } else {
5846 tempbx += tempcx;
5847 if(HwInfo->jChipType < SIS_315H) tempbx++;
5848 else if(issis) tempbx++;
5849 }
5850
5851 if(tempbx >= SiS_Pr->SiS_VT) tempbx -= SiS_Pr->SiS_VT; /* BPLVRS */
5852
5853 temp = tempbx & 0x00FF;
5854 if(SiS_Pr->SiS_IF_DEF_TRUMPION) {
5855 if(SiS_Pr->SiS_LCDResInfo == Panel_1024x768) {
5856 if(ModeNo == 0x10) temp = 0xa9;
5857 }
5858 }
5859 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x18,temp);
5860
5861 tempcx >>= 3;
5862 tempcx++;
5863
5864 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA)) {
5865 if(!(SiS_Pr->SiS_LCDInfo & LCDPass11)) {
5866 if(SiS_Pr->PanelVRE != 999) tempcx = SiS_Pr->PanelVRE;
5867 }
5868 }
5869
5870 tempcx += tempbx;
5871 temp = tempcx & 0x000F;
5872 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x19,0xF0,temp); /* BPLVRE */
5873
5874 temp = ((tempbx >> 8) & 0x07) << 3;
5875 if(SiS_Pr->SiS_IF_DEF_FSTN || SiS_Pr->SiS_IF_DEF_DSTN) {
5876 if(SiS_Pr->SiS_HDE != 640) {
5877 if(SiS_Pr->SiS_VGAVDE != SiS_Pr->SiS_VDE) temp |= 0x40;
5878 }
5879 } else if(SiS_Pr->SiS_VGAVDE != SiS_Pr->SiS_VDE) temp |= 0x40;
5880 if(SiS_Pr->SiS_SetFlag & EnableLVDSDDA) temp |= 0x40;
5881 tempbx = 0x87;
5882 if((HwInfo->jChipType >= SIS_315H) ||
5883 (HwInfo->jChipRevision >= 0x30)) {
5884 tempbx = 0x07;
5885 if((SiS_Pr->SiS_IF_DEF_CH70xx == 1) && (SiS_Pr->SiS_VBInfo & SetCRT2ToTV)) {
5886 if(SiS_GetReg(SiS_Pr->SiS_Part1Port,0x00) & 0x03) temp |= 0x80;
5887 }
5888 /* Chrontel 701x operates in 24bit mode (8-8-8, 2x12bit mutliplexed) via VGA2 */
5889 if(SiS_Pr->SiS_LCDInfo & LCDRGB18Bit) {
5890 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA) {
5891 if(SiS_GetReg(SiS_Pr->SiS_P3c4,0x06) & 0x10) temp |= 0x80;
5892 } else {
5893 if(SiS_GetReg(SiS_Pr->SiS_Part1Port,0x00) & 0x01) temp |= 0x80;
5894 }
5895 }
5896 }
5897 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x1A,tempbx,temp);
5898
5899 tempbx = push2; /* BPLVDEE */
5900
5901 tempcx = SiS_Pr->SiS_LCDVDES; /* BPLVDES */
5902
5903 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA)) {
5904 switch(SiS_Pr->SiS_LCDResInfo) {
5905 case Panel_640x480:
5906 tempbx = SiS_Pr->SiS_VGAVDE - 1;
5907 tempcx = SiS_Pr->SiS_VGAVDE;
5908 break;
5909 case Panel_800x600:
5910 if(!(SiS_Pr->SiS_VBInfo & SetInSlaveMode)) {
5911 if(resinfo == SIS_RI_800x600) tempcx++;
5912 }
5913 break;
5914 case Panel_1024x600:
5915 if(!(SiS_Pr->SiS_VBInfo & SetInSlaveMode)) {
5916 if(resinfo == SIS_RI_1024x600) tempcx++;
5917 if(SiS_Pr->SiS_LCDInfo & DontExpandLCD) {
5918 if(resinfo == SIS_RI_800x600) tempcx++;
5919 }
5920 }
5921 break;
5922 case Panel_1024x768:
5923 if(HwInfo->jChipType < SIS_315H) {
5924 if(!(SiS_Pr->SiS_VBInfo & SetInSlaveMode)) {
5925 if(resinfo == SIS_RI_1024x768) tempcx++;
5926 }
5927 }
5928 break;
5929 }
5930 }
5931
5932 temp = ((tempbx >> 8) & 0x07) << 3;
5933 temp = temp | ((tempcx >> 8) & 0x07);
5934 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1D,temp);
5935 /* if(SiS_Pr->SiS_IF_DEF_FSTN) tempbx++; */
5936 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1C,tempbx);
5937 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1B,tempcx);
5938
5939 /* Vertical scaling */
5940
5941 if(HwInfo->jChipType < SIS_315H) {
5942
5943#ifdef SIS300 /* 300 series */
5944 tempeax = SiS_Pr->SiS_VGAVDE << 6;
5945 temp = (tempeax % (ULONG)SiS_Pr->SiS_VDE);
5946 tempeax = tempeax / (ULONG)SiS_Pr->SiS_VDE;
5947 if(temp) tempeax++;
5948
5949 if(SiS_Pr->SiS_SetFlag & EnableLVDSDDA) tempeax = 0x3F;
5950
5951 temp = (USHORT)(tempeax & 0x00FF);
5952 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1E,temp); /* BPLVCFACT */
5953 tempvcfact = temp;
5954#endif /* SIS300 */
5955
5956 } else {
5957
5958#ifdef SIS315H /* 315 series */
5959 tempeax = SiS_Pr->SiS_VGAVDE << 18;
5960 tempebx = SiS_Pr->SiS_VDE;
5961 temp = (tempeax % tempebx);
5962 tempeax = tempeax / tempebx;
5963 if(temp) tempeax++;
5964 tempvcfact = tempeax;
5965
5966 temp = (USHORT)(tempeax & 0x00FF);
5967 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x37,temp);
5968 temp = (USHORT)((tempeax & 0x00FF00) >> 8);
5969 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x36,temp);
5970 temp = (USHORT)((tempeax & 0x00030000) >> 16);
5971 if(SiS_Pr->SiS_VDE == SiS_Pr->SiS_VGAVDE) temp |= 0x04;
5972 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x35,temp);
5973
5974 if(SiS_Pr->SiS_VBType & (VB_SIS301C | VB_SIS302ELV)) {
5975 temp = (USHORT)(tempeax & 0x00FF);
5976 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x3c,temp);
5977 temp = (USHORT)((tempeax & 0x00FF00) >> 8);
5978 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x3b,temp);
5979 temp = (USHORT)(((tempeax & 0x00030000) >> 16) << 6);
5980 SiS_SetRegANDOR(SiS_Pr->SiS_Part4Port,0x3a,0x3f,temp);
5981 temp = 0;
5982 if(SiS_Pr->SiS_VDE != SiS_Pr->SiS_VGAVDE) temp |= 0x08;
5983 SiS_SetRegANDOR(SiS_Pr->SiS_Part4Port,0x30,0xf3,temp);
5984 }
5985#endif
5986
5987 }
5988
5989 /* Horizontal scaling */
5990
5991 tempeax = SiS_Pr->SiS_VGAHDE; /* 1f = ( (VGAHDE * 65536) / ( (VGAHDE * 65536) / HDE ) ) - 1*/
5992 if(chkdclkfirst) {
5993 if(modeflag & HalfDCLK) tempeax >>= 1;
5994 }
5995 tempebx = tempeax << 16;
5996 if(SiS_Pr->SiS_HDE == tempeax) {
5997 tempecx = 0xFFFF;
5998 } else {
5999 tempecx = tempebx / SiS_Pr->SiS_HDE;
6000 if(HwInfo->jChipType >= SIS_315H) {
6001 if(tempebx % SiS_Pr->SiS_HDE) tempecx++;
6002 }
6003 }
6004
6005 if(HwInfo->jChipType >= SIS_315H) {
6006 tempeax = (tempebx / tempecx) - 1;
6007 } else {
6008 tempeax = ((SiS_Pr->SiS_VGAHT << 16) / tempecx) - 1;
6009 }
6010 tempecx = (tempecx << 16) | (tempeax & 0xFFFF);
6011 temp = (USHORT)(tempecx & 0x00FF);
6012 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1F,temp);
6013
6014 if(HwInfo->jChipType >= SIS_315H) {
6015 tempeax = (SiS_Pr->SiS_VGAVDE << 18) / tempvcfact;
6016 tempbx = (USHORT)(tempeax & 0xFFFF);
6017 } else {
6018 tempeax = SiS_Pr->SiS_VGAVDE << 6;
6019 tempbx = tempvcfact & 0x3f;
6020 if(tempbx == 0) tempbx = 64;
6021 tempeax /= tempbx;
6022 tempbx = (USHORT)(tempeax & 0xFFFF);
6023 }
6024 if(SiS_Pr->SiS_LCDResInfo == Panel_1024x768) tempbx--;
6025 if(SiS_Pr->SiS_SetFlag & EnableLVDSDDA) {
6026 if((!SiS_Pr->SiS_IF_DEF_FSTN) && (!SiS_Pr->SiS_IF_DEF_DSTN)) tempbx = 1;
6027 else if(SiS_Pr->SiS_LCDResInfo != Panel_640x480) tempbx = 1;
6028 }
6029
6030 temp = ((tempbx >> 8) & 0x07) << 3;
6031 temp = temp | ((tempecx >> 8) & 0x07);
6032 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x20,temp);
6033 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x21,tempbx);
6034
6035 tempecx >>= 16; /* BPLHCFACT */
6036 if(!chkdclkfirst) {
6037 if(modeflag & HalfDCLK) tempecx >>= 1;
6038 }
6039 temp = (USHORT)((tempecx & 0xFF00) >> 8);
6040 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x22,temp);
6041 temp = (USHORT)(tempecx & 0x00FF);
6042 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x23,temp);
6043
6044#ifdef SIS315H
6045 if(HwInfo->jChipType >= SIS_315H) {
6046 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA) {
6047 if((islvds) || (SiS_Pr->SiS_VBInfo & VB_SIS301LV302LV)) {
6048 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1e,0x20);
6049 }
6050 } else {
6051 if(islvds) {
6052 if(HwInfo->jChipType == SIS_740) {
6053 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x1e,0x03);
6054 } else {
6055 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1e,0x23);
6056 }
6057 }
6058 }
6059 }
6060#endif
6061
6062#ifdef SIS300
6063 if(SiS_Pr->SiS_IF_DEF_TRUMPION) {
6064 int i;
6065 UCHAR TrumpMode13[4] = { 0x01, 0x10, 0x2c, 0x00 };
6066 UCHAR TrumpMode10_1[4] = { 0x01, 0x10, 0x27, 0x00 };
6067 UCHAR TrumpMode10_2[4] = { 0x01, 0x16, 0x10, 0x00 };
6068
6069 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x02,0xbf);
6070 for(i=0; i<5; i++) {
6071 SiS_SetTrumpionBlock(SiS_Pr, &SiS300_TrumpionData[crt2crtc][0]);
6072 }
6073 if(SiS_Pr->SiS_LCDResInfo == Panel_1024x768) {
6074 if(ModeNo == 0x13) {
6075 for(i=0; i<4; i++) {
6076 SiS_SetTrumpionBlock(SiS_Pr, &TrumpMode13[0]);
6077 }
6078 } else if(ModeNo == 0x10) {
6079 for(i=0; i<4; i++) {
6080 SiS_SetTrumpionBlock(SiS_Pr, &TrumpMode10_1[0]);
6081 SiS_SetTrumpionBlock(SiS_Pr, &TrumpMode10_2[0]);
6082 }
6083 }
6084 }
6085 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x02,0x40);
6086 }
6087#endif
6088
6089#ifdef SIS315H
6090 if(SiS_Pr->SiS_IF_DEF_FSTN || SiS_Pr->SiS_IF_DEF_DSTN) {
6091 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x25,0x00);
6092 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x26,0x00);
6093 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x27,0x00);
6094 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x28,0x87);
6095 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x29,0x5A);
6096 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x2A,0x4B);
6097 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x44,~0x07,0x03);
6098 tempax = SiS_Pr->SiS_HDE; /* Blps = lcdhdee(lcdhdes+HDE) + 64 */
6099 if(SiS_Pr->SiS_LCDResInfo == Panel_640x480_2 ||
6100 SiS_Pr->SiS_LCDResInfo == Panel_640x480_3) tempax >>= 1;
6101 tempax += 64;
6102 temp = tempax & 0x00FF;
6103 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x38,temp);
6104 temp = ((tempax & 0xFF00) >> 8) << 3;
6105 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x35,~0x078,temp);
6106 tempax += 32; /* Blpe=lBlps+32 */
6107 temp = tempax & 0x00FF;
6108 if(SiS_Pr->SiS_IF_DEF_FSTN) temp = 0;
6109 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x39,temp);
6110 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x3A,0x00); /* Bflml=0 */
6111 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x3C,~0x007,0x00);
6112
6113 tempax = SiS_Pr->SiS_VDE;
6114 if(SiS_Pr->SiS_LCDResInfo == Panel_640x480_2 ||
6115 SiS_Pr->SiS_LCDResInfo == Panel_640x480_3) tempax >>= 1;
6116 tempax >>= 1;
6117 temp = tempax & 0x00FF;
6118 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x3B,temp);
6119 temp = ((tempax & 0xFF00) >> 8) << 3;
6120 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x3C,~0x038,temp);
6121
6122 tempeax = SiS_Pr->SiS_HDE;
6123 if(SiS_Pr->SiS_LCDResInfo == Panel_640x480_2 ||
6124 SiS_Pr->SiS_LCDResInfo == Panel_640x480_3) tempeax >>= 1;
6125 tempeax <<= 2; /* BDxFIFOSTOP = (HDE*4)/128 */
6126 tempebx = 128;
6127 temp = (USHORT)(tempeax % tempebx);
6128 tempeax = tempeax / tempebx;
6129 if(temp) tempeax++;
6130 temp = (USHORT)(tempeax & 0x003F);
6131 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x45,~0x0FF,temp);
6132 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x3F,0x00); /* BDxWadrst0 */
6133 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x3E,0x00);
6134 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x3D,0x10);
6135 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x3C,~0x040,0x00);
6136
6137 tempax = SiS_Pr->SiS_HDE;
6138 if(SiS_Pr->SiS_LCDResInfo == Panel_640x480_2 ||
6139 SiS_Pr->SiS_LCDResInfo == Panel_640x480_3) tempax >>= 1;
6140 tempax >>= 4; /* BDxWadroff = HDE*4/8/8 */
6141 pushcx = tempax;
6142 temp = tempax & 0x00FF;
6143 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x43,temp);
6144 temp = ((tempax & 0xFF00) >> 8) << 3;
6145 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x44,~0x0F8,temp);
6146
6147 tempax = SiS_Pr->SiS_VDE; /* BDxWadrst1 = BDxWadrst0 + BDxWadroff * VDE */
6148 if(SiS_Pr->SiS_LCDResInfo == Panel_640x480_2 ||
6149 SiS_Pr->SiS_LCDResInfo == Panel_640x480_3) tempax >>= 1;
6150 tempeax = (tempax * pushcx);
6151 tempebx = 0x00100000 + tempeax;
6152 temp = (USHORT)tempebx & 0x000000FF;
6153 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x42,temp);
6154 temp = (USHORT)((tempebx & 0x0000FF00) >> 8);
6155 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x41,temp);
6156 temp = (USHORT)((tempebx & 0x00FF0000) >> 16);
6157 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x40,temp);
6158 temp = (USHORT)(((tempebx & 0x01000000) >> 24) << 7);
6159 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x3C,~0x080,temp);
6160
6161 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x2F,0x03);
6162 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x03,0x50);
6163 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x04,0x00);
6164 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x2F,0x01);
6165 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x19,0x38);
6166
6167 if(SiS_Pr->SiS_IF_DEF_FSTN) {
6168 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x2b,0x02);
6169 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x2c,0x00);
6170 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x2d,0x00);
6171 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x35,0x0c);
6172 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x36,0x00);
6173 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x37,0x00);
6174 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x38,0x80);
6175 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x39,0xA0);
6176 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x3a,0x00);
6177 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x3b,0xf0);
6178 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x3c,0x00);
6179 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x3d,0x10);
6180 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x3e,0x00);
6181 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x3f,0x00);
6182 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x40,0x10);
6183 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x41,0x25);
6184 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x42,0x80);
6185 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x43,0x14);
6186 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x44,0x03);
6187 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x45,0x0a);
6188 }
6189 }
6190#endif /* SIS315H */
6191}
6192
6193/* Set Part 1 */
6194static void
6195SiS_SetGroup1(SiS_Private *SiS_Pr, USHORT ModeNo, USHORT ModeIdIndex,
6196 PSIS_HW_INFO HwInfo, USHORT RefreshRateTableIndex)
6197{
6198#if defined(SIS300) || defined(SIS315H)
6199 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
6200#endif
6201 USHORT temp=0, tempax=0, tempbx=0, tempcx=0, bridgeadd=0;
6202 USHORT pushbx=0, CRT1Index=0, modeflag, resinfo=0;
6203#ifdef SIS315H
6204 USHORT tempbl=0;
6205#endif
6206
6207 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA) {
6208 SiS_SetGroup1_LVDS(SiS_Pr, ModeNo, ModeIdIndex, HwInfo, RefreshRateTableIndex);
6209 return;
6210 }
6211
6212 if(ModeNo <= 0x13) {
6213 modeflag = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_ModeFlag;
6214 } else if(SiS_Pr->UseCustomMode) {
6215 modeflag = SiS_Pr->CModeFlag;
6216 } else {
6217 CRT1Index = SiS_Pr->SiS_RefIndex[RefreshRateTableIndex].Ext_CRT1CRTC;
6218 resinfo = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_RESINFO;
6219 modeflag = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_ModeFlag;
6220 }
6221
6222 SiS_SetCRT2Offset(SiS_Pr, ModeNo, ModeIdIndex, RefreshRateTableIndex, HwInfo);
6223
6224 if( ! ((HwInfo->jChipType >= SIS_315H) &&
6225 (SiS_Pr->SiS_IF_DEF_LVDS == 1) &&
6226 (SiS_Pr->SiS_VBInfo & SetInSlaveMode)) ) {
6227
6228 if(HwInfo->jChipType < SIS_315H ) {
6229#ifdef SIS300
6230 SiS_SetCRT2FIFO_300(SiS_Pr, ModeNo, HwInfo);
6231#endif
6232 } else {
6233#ifdef SIS315H
6234 SiS_SetCRT2FIFO_310(SiS_Pr, HwInfo);
6235#endif
6236 }
6237
6238 /* 1. Horizontal setup */
6239
6240 if(HwInfo->jChipType < SIS_315H ) {
6241
6242#ifdef SIS300 /* ------------- 300 series --------------*/
6243
6244 temp = (SiS_Pr->SiS_VGAHT - 1) & 0x0FF; /* BTVGA2HT 0x08,0x09 */
6245 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x08,temp); /* CRT2 Horizontal Total */
6246
6247 temp = (((SiS_Pr->SiS_VGAHT - 1) & 0xFF00) >> 8) << 4;
6248 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x09,0x0f,temp); /* CRT2 Horizontal Total Overflow [7:4] */
6249
6250 temp = (SiS_Pr->SiS_VGAHDE + 12) & 0x0FF; /* BTVGA2HDEE 0x0A,0x0C */
6251 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x0A,temp); /* CRT2 Horizontal Display Enable End */
6252
6253 pushbx = SiS_Pr->SiS_VGAHDE + 12; /* bx BTVGA2HRS 0x0B,0x0C */
6254 tempcx = (SiS_Pr->SiS_VGAHT - SiS_Pr->SiS_VGAHDE) >> 2;
6255 tempbx = pushbx + tempcx;
6256 tempcx <<= 1;
6257 tempcx += tempbx;
6258
6259 bridgeadd = 12;
6260
6261#endif /* SIS300 */
6262
6263 } else {
6264
6265#ifdef SIS315H /* ------------------- 315/330 series --------------- */
6266
6267 tempcx = SiS_Pr->SiS_VGAHT; /* BTVGA2HT 0x08,0x09 */
6268 if(modeflag & HalfDCLK) {
6269 if(SiS_Pr->SiS_VBType & VB_SISVB) {
6270 tempcx >>= 1;
6271 } else {
6272 tempax = SiS_Pr->SiS_VGAHDE >> 1;
6273 tempcx = SiS_Pr->SiS_HT - SiS_Pr->SiS_HDE + tempax;
6274 if(SiS_Pr->SiS_LCDInfo & DontExpandLCD) {
6275 tempcx = SiS_Pr->SiS_HT - tempax;
6276 }
6277 }
6278 }
6279 tempcx--;
6280 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x08,tempcx); /* CRT2 Horizontal Total */
6281 temp = (tempcx >> 4) & 0xF0;
6282 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x09,0x0F,temp); /* CRT2 Horizontal Total Overflow [7:4] */
6283
6284 tempcx = SiS_Pr->SiS_VGAHT; /* BTVGA2HDEE 0x0A,0x0C */
6285 tempbx = SiS_Pr->SiS_VGAHDE;
6286 tempcx -= tempbx;
6287 tempcx >>= 2;
6288 if(modeflag & HalfDCLK) {
6289 tempbx >>= 1;
6290 tempcx >>= 1;
6291 }
6292 tempbx += 16;
6293
6294 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x0A,tempbx); /* CRT2 Horizontal Display Enable End */
6295
6296 pushbx = tempbx;
6297 tempcx >>= 1;
6298 tempbx += tempcx;
6299 tempcx += tempbx;
6300
6301 bridgeadd = 16;
6302
6303 if(SiS_Pr->SiS_VBType & VB_SISVB) {
6304 if(HwInfo->jChipType >= SIS_661) {
6305 if((SiS_Pr->SiS_LCDResInfo == Panel_1600x1200) ||
6306 (SiS_Pr->SiS_LCDResInfo == Panel_1280x1024)) {
6307 if(resinfo == SIS_RI_1280x1024) {
6308 tempcx = (tempcx & 0xff00) | 0x30;
6309 } else if(resinfo == SIS_RI_1600x1200) {
6310 tempcx = (tempcx & 0xff00) | 0xff;
6311 }
6312 }
6313 }
6314 }
6315
6316#endif /* SIS315H */
6317
6318 } /* 315/330 series */
6319
6320 if(SiS_Pr->SiS_VBType & VB_SISVB) {
6321
6322 if(SiS_Pr->UseCustomMode) {
6323 tempbx = SiS_Pr->CHSyncStart + bridgeadd;
6324 tempcx = SiS_Pr->CHSyncEnd + bridgeadd;
6325 tempax = SiS_Pr->SiS_VGAHT;
6326 if(modeflag & HalfDCLK) tempax >>= 1;
6327 tempax--;
6328 if(tempcx > tempax) tempcx = tempax;
6329 }
6330
6331 if(SiS_Pr->SiS_VBInfo & SetCRT2ToRAMDAC) {
6332 unsigned char cr4, cr14, cr5, cr15;
6333 if(SiS_Pr->UseCustomMode) {
6334 cr4 = SiS_Pr->CCRT1CRTC[4];
6335 cr14 = SiS_Pr->CCRT1CRTC[14];
6336 cr5 = SiS_Pr->CCRT1CRTC[5];
6337 cr15 = SiS_Pr->CCRT1CRTC[15];
6338 } else {
6339 cr4 = SiS_Pr->SiS_CRT1Table[CRT1Index].CR[4];
6340 cr14 = SiS_Pr->SiS_CRT1Table[CRT1Index].CR[14];
6341 cr5 = SiS_Pr->SiS_CRT1Table[CRT1Index].CR[5];
6342 cr15 = SiS_Pr->SiS_CRT1Table[CRT1Index].CR[15];
6343 }
6344 tempbx = ((cr4 | ((cr14 & 0xC0) << 2)) - 3) << 3; /* (VGAHRS-3)*8 */
6345 tempcx = (((cr5 & 0x1f) | ((cr15 & 0x04) << (5-2))) - 3) << 3; /* (VGAHRE-3)*8 */
6346 tempcx &= 0x00FF;
6347 tempcx |= (tempbx & 0xFF00);
6348 tempbx += bridgeadd;
6349 tempcx += bridgeadd;
6350 tempax = SiS_Pr->SiS_VGAHT;
6351 if(modeflag & HalfDCLK) tempax >>= 1;
6352 tempax--;
6353 if(tempcx > tempax) tempcx = tempax;
6354 }
6355
6356 if(SiS_Pr->SiS_TVMode & TVSetNTSC1024) {
6357 tempbx = 1040;
6358 tempcx = 1044; /* HWCursor bug! */
6359 }
6360
6361 }
6362
6363 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x0B,tempbx); /* CRT2 Horizontal Retrace Start */
6364
6365 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x0D,tempcx); /* CRT2 Horizontal Retrace End */
6366
6367 temp = ((tempbx >> 8) & 0x0F) | ((pushbx >> 4) & 0xF0);
6368 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x0C,temp); /* Overflow */
6369
6370 /* 2. Vertical setup */
6371
6372 tempcx = SiS_Pr->SiS_VGAVT - 1;
6373 temp = tempcx & 0x00FF;
6374
6375 if(HwInfo->jChipType < SIS_661) {
6376 if(SiS_Pr->SiS_IF_DEF_LVDS == 1) {
6377 if(HwInfo->jChipType < SIS_315H) {
6378 if(SiS_Pr->SiS_IF_DEF_CH70xx != 0) {
6379 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToSVIDEO | SetCRT2ToAVIDEO)) {
6380 temp--;
6381 }
6382 }
6383 } else {
6384 temp--;
6385 }
6386 } else if(HwInfo->jChipType >= SIS_315H) {
6387 temp--;
6388 }
6389 }
6390 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x0E,temp); /* CRT2 Vertical Total */
6391
6392 tempbx = SiS_Pr->SiS_VGAVDE - 1;
6393 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x0F,tempbx); /* CRT2 Vertical Display Enable End */
6394
6395 temp = ((tempbx >> 5) & 0x38) | ((tempcx >> 8) & 0x07);
6396 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x12,temp); /* Overflow */
6397
6398 if((HwInfo->jChipType >= SIS_315H) && (HwInfo->jChipType < SIS_661)) {
6399 tempbx++;
6400 tempax = tempbx;
6401 tempcx++;
6402 tempcx -= tempax;
6403 tempcx >>= 2;
6404 tempbx += tempcx;
6405 if(tempcx < 4) tempcx = 4;
6406 tempcx >>= 2;
6407 tempcx += tempbx;
6408 tempcx++;
6409 } else {
6410 tempbx = (SiS_Pr->SiS_VGAVT + SiS_Pr->SiS_VGAVDE) >> 1; /* BTVGA2VRS 0x10,0x11 */
6411 tempcx = ((SiS_Pr->SiS_VGAVT - SiS_Pr->SiS_VGAVDE) >> 4) + tempbx + 1; /* BTVGA2VRE 0x11 */
6412 }
6413
6414 if(SiS_Pr->SiS_VBType & VB_SISVB) {
6415 if(SiS_Pr->UseCustomMode) {
6416 tempbx = SiS_Pr->CVSyncStart;
6417 tempcx = SiS_Pr->CVSyncEnd;
6418 }
6419 if(SiS_Pr->SiS_VBInfo & SetCRT2ToRAMDAC) {
6420 unsigned char cr8, cr7, cr13;
6421 if(SiS_Pr->UseCustomMode) {
6422 cr8 = SiS_Pr->CCRT1CRTC[8];
6423 cr7 = SiS_Pr->CCRT1CRTC[7];
6424 cr13 = SiS_Pr->CCRT1CRTC[13];
6425 tempcx = SiS_Pr->CCRT1CRTC[9];
6426 } else {
6427 cr8 = SiS_Pr->SiS_CRT1Table[CRT1Index].CR[8];
6428 cr7 = SiS_Pr->SiS_CRT1Table[CRT1Index].CR[7];
6429 cr13 = SiS_Pr->SiS_CRT1Table[CRT1Index].CR[13];
6430 tempcx = SiS_Pr->SiS_CRT1Table[CRT1Index].CR[9];
6431 }
6432 tempbx = cr8;
6433 if(cr7 & 0x04) tempbx |= 0x0100;
6434 if(cr7 & 0x80) tempbx |= 0x0200;
6435 if(cr13 & 0x08) tempbx |= 0x0400;
6436 }
6437 }
6438 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x10,tempbx); /* CRT2 Vertical Retrace Start */
6439
6440 temp = ((tempbx >> 4) & 0x70) | (tempcx & 0x0F);
6441 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x11,temp); /* CRT2 Vert. Retrace End; Overflow */
6442
6443 /* 3. Panel delay compensation */
6444
6445 if(HwInfo->jChipType < SIS_315H) {
6446
6447#ifdef SIS300 /* ---------- 300 series -------------- */
6448
6449 if(SiS_Pr->SiS_VBType & VB_SISVB) {
6450 temp = 0x20;
6451 if(HwInfo->jChipType == SIS_300) {
6452 temp = 0x10;
6453 if(SiS_Pr->SiS_LCDResInfo == Panel_1024x768) temp = 0x2c;
6454 if(SiS_Pr->SiS_LCDResInfo == Panel_1280x1024) temp = 0x20;
6455 }
6456 if(SiS_Pr->SiS_VBType & VB_SIS301) {
6457 if(SiS_Pr->SiS_LCDResInfo == Panel_1280x1024) temp = 0x20;
6458 }
6459 if(SiS_Pr->SiS_LCDResInfo == Panel_1280x960) temp = 0x24;
6460 if(SiS_Pr->SiS_LCDResInfo == Panel_Custom) temp = 0x2c;
6461 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) temp = 0x08;
6462 if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) {
6463 if(SiS_Pr->SiS_VBInfo & SetInSlaveMode) temp = 0x2c;
6464 else temp = 0x20;
6465 }
6466 if(SiS_Pr->SiS_UseROM) {
6467 if(ROMAddr[0x220] & 0x80) {
6468 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTVNoYPbPrHiVision)
6469 temp = ROMAddr[0x221];
6470 else if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision)
6471 temp = ROMAddr[0x222];
6472 else if(SiS_Pr->SiS_LCDResInfo == Panel_1280x1024)
6473 temp = ROMAddr[0x223];
6474 else
6475 temp = ROMAddr[0x224];
6476 temp &= 0x3c;
6477 }
6478 }
6479 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) {
6480 if(SiS_Pr->PDC != -1) temp = SiS_Pr->PDC & 0x3c;
6481 }
6482
6483 } else {
6484 temp = 0x20;
6485 if(!(SiS_Pr->SiS_VBInfo & SetCRT2ToTV)) {
6486 if(SiS_Pr->SiS_LCDResInfo == Panel_640x480) temp = 0x04;
6487 }
6488 if(SiS_Pr->SiS_UseROM) {
6489 if(ROMAddr[0x220] & 0x80) {
6490 temp = ROMAddr[0x220] & 0x3c;
6491 }
6492 }
6493 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) {
6494 if(SiS_Pr->PDC != -1) temp = SiS_Pr->PDC & 0x3c;
6495 }
6496 }
6497
6498 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x13,~0x3C,temp); /* Panel Link Delay Compensation; (Software Command Reset; Power Saving) */
6499
6500#endif /* SIS300 */
6501
6502 } else {
6503
6504#ifdef SIS315H /* --------------- 315/330 series ---------------*/
6505
6506 if(HwInfo->jChipType < SIS_661) {
6507
6508 if(SiS_Pr->SiS_IF_DEF_LVDS == 1) {
6509
6510 if(HwInfo->jChipType == SIS_740) temp = 0x03;
6511 else temp = 0x00;
6512
6513 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) temp = 0x0a;
6514 tempbl = 0xF0;
6515 if(HwInfo->jChipType == SIS_650) {
6516 if(SiS_Pr->SiS_IF_DEF_CH70xx != 0) {
6517 if(!(SiS_Pr->SiS_VBInfo & SetCRT2ToTV)) tempbl = 0x0F;
6518 }
6519 }
6520
6521 if(SiS_Pr->SiS_IF_DEF_DSTN || SiS_Pr->SiS_IF_DEF_FSTN) {
6522 temp = 0x08;
6523 tempbl = 0;
6524 if((SiS_Pr->SiS_UseROM) && (!(SiS_Pr->SiS_ROMNew))) {
6525 if(ROMAddr[0x13c] & 0x80) tempbl = 0xf0;
6526 }
6527 }
6528
6529 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x2D,tempbl,temp); /* Panel Link Delay Compensation */
6530 }
6531
6532 } /* < 661 */
6533
6534 tempax = 0;
6535 if(modeflag & DoubleScanMode) tempax |= 0x80;
6536 if(modeflag & HalfDCLK) tempax |= 0x40;
6537 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x2C,0x3f,tempax);
6538
6539#endif /* SIS315H */
6540
6541 }
6542
6543 } /* Slavemode */
6544
6545 if(SiS_Pr->SiS_VBType & VB_SISVB) {
6546 if((SiS_Pr->SiS_VBType & VB_NoLCD) && (SiS_Pr->SiS_VBInfo & SetCRT2ToLCD)) {
6547 /* For 301BDH with LCD, we set up the Panel Link */
6548 SiS_SetGroup1_LVDS(SiS_Pr, ModeNo, ModeIdIndex, HwInfo, RefreshRateTableIndex);
6549 } else if(SiS_Pr->SiS_VBInfo & SetInSlaveMode) {
6550 SiS_SetGroup1_301(SiS_Pr, ModeNo, ModeIdIndex, HwInfo, RefreshRateTableIndex);
6551 }
6552 } else {
6553 if(HwInfo->jChipType < SIS_315H) {
6554 SiS_SetGroup1_LVDS(SiS_Pr, ModeNo, ModeIdIndex, HwInfo, RefreshRateTableIndex);
6555 } else {
6556 if(SiS_Pr->SiS_IF_DEF_CH70xx != 0) {
6557 if((!(SiS_Pr->SiS_VBInfo & SetCRT2ToTV)) || (SiS_Pr->SiS_VBInfo & SetInSlaveMode)) {
6558 SiS_SetGroup1_LVDS(SiS_Pr, ModeNo,ModeIdIndex, HwInfo,RefreshRateTableIndex);
6559 }
6560 } else {
6561 SiS_SetGroup1_LVDS(SiS_Pr, ModeNo,ModeIdIndex, HwInfo,RefreshRateTableIndex);
6562 }
6563 }
6564 }
6565}
6566
6567/*********************************************/
6568/* SET PART 2 REGISTER GROUP */
6569/*********************************************/
6570
6571#ifdef SIS315H
6572static UCHAR *
6573SiS_GetGroup2CLVXPtr(SiS_Private *SiS_Pr, int tabletype, PSIS_HW_INFO HwInfo)
6574{
6575 const UCHAR *tableptr = NULL;
6576 USHORT a, b, p = 0;
6577
6578 a = SiS_Pr->SiS_VGAHDE;
6579 b = SiS_Pr->SiS_HDE;
6580 if(tabletype) {
6581 a = SiS_Pr->SiS_VGAVDE;
6582 b = SiS_Pr->SiS_VDE;
6583 }
6584
6585 if(a < b) {
6586 tableptr = SiS_Part2CLVX_1;
6587 } else if(a == b) {
6588 tableptr = SiS_Part2CLVX_2;
6589 } else {
6590 if(SiS_Pr->SiS_TVMode & TVSetPAL) {
6591 tableptr = SiS_Part2CLVX_4;
6592 } else {
6593 tableptr = SiS_Part2CLVX_3;
6594 }
6595 if(SiS_Pr->SiS_VBInfo & SetCRT2ToYPbPr525750) {
6596 if(SiS_Pr->SiS_TVMode & TVSetYPbPr525i) tableptr = SiS_Part2CLVX_3;
6597 else if(SiS_Pr->SiS_TVMode & TVSetYPbPr525p) tableptr = SiS_Part2CLVX_3;
6598 else tableptr = SiS_Part2CLVX_5;
6599 } else if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) {
6600 tableptr = SiS_Part2CLVX_6;
6601 }
6602 do {
6603 if((tableptr[p] | tableptr[p+1] << 8) == a) break;
6604 p += 0x42;
6605 } while((tableptr[p] | tableptr[p+1] << 8) != 0xffff);
6606 if((tableptr[p] | tableptr[p+1] << 8) == 0xffff) p -= 0x42;
6607 }
6608 p += 2;
6609 return((UCHAR *)&tableptr[p]);
6610}
6611
6612static void
6613SiS_SetGroup2_C_ELV(SiS_Private *SiS_Pr, USHORT ModeNo, USHORT ModeIdIndex,
6614 USHORT RefreshRateTableIndex, PSIS_HW_INFO HwInfo)
6615{
6616 UCHAR *tableptr;
6617 int i, j;
6618 UCHAR temp;
6619
6620 if(!(SiS_Pr->SiS_VBType & (VB_SIS301C | VB_SIS302ELV))) return;
6621
6622 tableptr = SiS_GetGroup2CLVXPtr(SiS_Pr, 0, HwInfo);
6623 for(i = 0x80, j = 0; i <= 0xbf; i++, j++) {
6624 SiS_SetReg(SiS_Pr->SiS_Part2Port, i, tableptr[j]);
6625 }
6626 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) {
6627 tableptr = SiS_GetGroup2CLVXPtr(SiS_Pr, 1, HwInfo);
6628 for(i = 0xc0, j = 0; i <= 0xff; i++, j++) {
6629 SiS_SetReg(SiS_Pr->SiS_Part2Port, i, tableptr[j]);
6630 }
6631 }
6632 temp = 0x10;
6633 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) temp |= 0x04;
6634 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x4e,0xeb,temp);
6635}
6636
6637static BOOLEAN
6638SiS_GetCRT2Part2Ptr(SiS_Private *SiS_Pr,USHORT ModeNo,USHORT ModeIdIndex,
6639 USHORT RefreshRateTableIndex,USHORT *CRT2Index,
6640 USHORT *ResIndex,PSIS_HW_INFO HwInfo)
6641{
6642
6643 if(HwInfo->jChipType < SIS_315H) return FALSE;
6644
6645 if(ModeNo <= 0x13)
6646 (*ResIndex) = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_CRT2CRTC;
6647 else
6648 (*ResIndex) = SiS_Pr->SiS_RefIndex[RefreshRateTableIndex].Ext_CRT2CRTC;
6649
6650 (*ResIndex) &= 0x3f;
6651 (*CRT2Index) = 0;
6652
6653 if(SiS_Pr->SiS_LCDResInfo == Panel_1024x768) {
6654 if(!(SiS_Pr->SiS_SetFlag & LCDVESATiming)) {
6655 (*CRT2Index) = 200;
6656 }
6657 }
6658
6659 if(SiS_Pr->SiS_CustomT == CUT_ASUSA2H_2) {
6660 if(SiS_Pr->SiS_LCDResInfo == Panel_1024x768) {
6661 if(SiS_Pr->SiS_SetFlag & LCDVESATiming) (*CRT2Index) = 206;
6662 }
6663 }
6664 return(((*CRT2Index) != 0));
6665}
6666#endif
6667
6668#ifdef SIS300
6669static void
6670SiS_Group2LCDSpecial(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo, USHORT ModeNo, USHORT crt2crtc)
6671{
6672 USHORT tempcx;
6673 const UCHAR atable[] = {
6674 0xc3,0x9e,0xc3,0x9e,0x02,0x02,0x02,
6675 0xab,0x87,0xab,0x9e,0xe7,0x02,0x02
6676 };
6677
6678 if(!SiS_Pr->UseCustomMode) {
6679 if( ( ( (HwInfo->jChipType == SIS_630) ||
6680 (HwInfo->jChipType == SIS_730) ) &&
6681 (HwInfo->jChipRevision > 2) ) &&
6682 (SiS_Pr->SiS_LCDResInfo == Panel_1024x768) &&
6683 (!(SiS_Pr->SiS_SetFlag & LCDVESATiming)) &&
6684 (!(SiS_Pr->SiS_LCDInfo & DontExpandLCD)) ) {
6685 if(ModeNo == 0x13) {
6686 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x04,0xB9);
6687 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x05,0xCC);
6688 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x06,0xA6);
6689 } else {
6690 if((crt2crtc & 0x3F) == 4) {
6691 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x01,0x2B);
6692 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x02,0x13);
6693 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x04,0xE5);
6694 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x05,0x08);
6695 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x06,0xE2);
6696 }
6697 }
6698 }
6699
6700 if(HwInfo->jChipType < SIS_315H) {
6701 if(SiS_Pr->SiS_LCDTypeInfo == 0x0c) {
6702 crt2crtc &= 0x1f;
6703 tempcx = 0;
6704 if(!(SiS_Pr->SiS_VBInfo & SetNotSimuMode)) {
6705 if(SiS_Pr->SiS_VBInfo & SetInSlaveMode) {
6706 tempcx += 7;
6707 }
6708 }
6709 tempcx += crt2crtc;
6710 if(crt2crtc >= 4) {
6711 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x06,0xff);
6712 }
6713
6714 if(!(SiS_Pr->SiS_VBInfo & SetNotSimuMode)) {
6715 if(SiS_Pr->SiS_VBInfo & SetInSlaveMode) {
6716 if(crt2crtc == 4) {
6717 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x01,0x28);
6718 }
6719 }
6720 }
6721 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x02,0x18);
6722 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x04,atable[tempcx]);
6723 }
6724 }
6725 }
6726}
6727
6728/* For ECS A907. Highly preliminary. */
6729static void
6730SiS_Set300Part2Regs(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo,
6731 USHORT ModeIdIndex, USHORT RefreshRateTableIndex,
6732 USHORT ModeNo)
6733{
6734 USHORT crt2crtc, resindex;
6735 int i,j;
6736 const SiS_Part2PortTblStruct *CRT2Part2Ptr = NULL;
6737
6738 if(HwInfo->jChipType != SIS_300) return;
6739 if(!(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV)) return;
6740 if(SiS_Pr->UseCustomMode) return;
6741
6742 if(ModeNo <= 0x13) {
6743 crt2crtc = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_CRT2CRTC;
6744 } else {
6745 crt2crtc = SiS_Pr->SiS_RefIndex[RefreshRateTableIndex].Ext_CRT2CRTC;
6746 }
6747
6748 resindex = crt2crtc & 0x3F;
6749 if(SiS_Pr->SiS_SetFlag & LCDVESATiming) CRT2Part2Ptr = SiS_Pr->SiS_CRT2Part2_1024x768_1;
6750 else CRT2Part2Ptr = SiS_Pr->SiS_CRT2Part2_1024x768_2;
6751
6752 /* The BIOS code (1.16.51,56) is obviously a fragment! */
6753 if(ModeNo > 0x13) {
6754 CRT2Part2Ptr = SiS_Pr->SiS_CRT2Part2_1024x768_1;
6755 resindex = 4;
6756 }
6757
6758 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x01,0x80,(CRT2Part2Ptr+resindex)->CR[0]);
6759 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x02,0x80,(CRT2Part2Ptr+resindex)->CR[1]);
6760 for(i = 2, j = 0x04; j <= 0x06; i++, j++ ) {
6761 SiS_SetReg(SiS_Pr->SiS_Part2Port,j,(CRT2Part2Ptr+resindex)->CR[i]);
6762 }
6763 for(j = 0x1c; j <= 0x1d; i++, j++ ) {
6764 SiS_SetReg(SiS_Pr->SiS_Part2Port,j,(CRT2Part2Ptr+resindex)->CR[i]);
6765 }
6766 for(j = 0x1f; j <= 0x21; i++, j++ ) {
6767 SiS_SetReg(SiS_Pr->SiS_Part2Port,j,(CRT2Part2Ptr+resindex)->CR[i]);
6768 }
6769 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x23,(CRT2Part2Ptr+resindex)->CR[10]);
6770 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x25,0x0f,(CRT2Part2Ptr+resindex)->CR[11]);
6771}
6772#endif
6773
6774static void
6775SiS_SetTVSpecial(SiS_Private *SiS_Pr, USHORT ModeNo)
6776{
6777 if(!(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV)) return;
6778 if(!(SiS_Pr->SiS_VBInfo & SetCRT2ToTVNoHiVision)) return;
6779 if(SiS_Pr->SiS_TVMode & (TVSetYPbPr525p | TVSetYPbPr750p)) return;
6780
6781 if(!(SiS_Pr->SiS_TVMode & TVSetPAL)) {
6782 if(SiS_Pr->SiS_TVMode & TVSetNTSC1024) {
6783 const UCHAR specialtv[] = {
6784 0xa7,0x07,0xf2,0x6e,0x17,0x8b,0x73,0x53,
6785 0x13,0x40,0x34,0xf4,0x63,0xbb,0xcc,0x7a,
6786 0x58,0xe4,0x73,0xda,0x13
6787 };
6788 int i, j;
6789 for(i = 0x1c, j = 0; i <= 0x30; i++, j++) {
6790 SiS_SetReg(SiS_Pr->SiS_Part2Port,i,specialtv[j]);
6791 }
6792 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x43,0x72);
6793 if(!(SiS_Pr->SiS_VBInfo & SetCRT2ToYPbPr525750)) {
6794 if(SiS_Pr->SiS_TVMode & TVSetPALM) {
6795 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x01,0x14);
6796 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x02,0x1b);
6797 } else {
6798 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x01,0x14); /* 15 */
6799 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x02,0x1a); /* 1b */
6800 }
6801 }
6802 }
6803 } else {
6804 if((ModeNo == 0x38) || (ModeNo == 0x4a) || (ModeNo == 0x64) ||
6805 (ModeNo == 0x52) || (ModeNo == 0x58) || (ModeNo == 0x5c)) {
6806 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x01,0x1b); /* 21 */
6807 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x02,0x54); /* 5a */
6808 } else {
6809 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x01,0x1a); /* 21 */
6810 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x02,0x53); /* 5a */
6811 }
6812 }
6813}
6814
6815static void
6816SiS_SetGroup2_Tail(SiS_Private *SiS_Pr, USHORT ModeNo)
6817{
6818 USHORT temp;
6819
6820 if(!(SiS_Pr->SiS_SetFlag & LCDVESATiming)) {
6821 if(SiS_Pr->SiS_VGAVDE == 525) {
6822 temp = 0xc3;
6823 if(SiS_Pr->SiS_ModeType <= ModeVGA) {
6824 temp++;
6825 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) temp += 2;
6826 }
6827 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x2f,temp);
6828 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x30,0xb3);
6829 } else if(SiS_Pr->SiS_VGAVDE == 420) {
6830 temp = 0x4d;
6831 if(SiS_Pr->SiS_ModeType <= ModeVGA) {
6832 temp++;
6833 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) temp++;
6834 }
6835 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x2f,temp);
6836 }
6837 }
6838
6839 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) {
6840 if(SiS_Pr->SiS_LCDResInfo == Panel_1400x1050) {
6841 if(SiS_Pr->SiS_VBType & VB_SIS301B302B) {
6842 SiS_SetRegOR(SiS_Pr->SiS_Part2Port,0x1a,0x03);
6843 /* Not always for LV, see SetGrp2 */
6844 }
6845 temp = 1;
6846 if(ModeNo <= 0x13) temp = 3;
6847 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x0b,temp);
6848 }
6849#if 0
6850 /* 651+301C, for 1280x768 - do I really need that? */
6851 if((SiS_Pr->SiS_PanelXRes == 1280) && (SiS_Pr->SiS_PanelYRes == 768)) {
6852 if(SiS_Pr->SiS_VBInfo & SetSimuScanMode) {
6853 if(((SiS_Pr->SiS_HDE == 640) && (SiS_Pr->SiS_VDE == 480)) ||
6854 ((SiS_Pr->SiS_HDE == 320) && (SiS_Pr->SiS_VDE == 240))) {
6855 SiS_SetReg(SiS_Part2Port,0x01,0x2b);
6856 SiS_SetReg(SiS_Part2Port,0x02,0x13);
6857 SiS_SetReg(SiS_Part2Port,0x04,0xe5);
6858 SiS_SetReg(SiS_Part2Port,0x05,0x08);
6859 SiS_SetReg(SiS_Part2Port,0x06,0xe2);
6860 SiS_SetReg(SiS_Part2Port,0x1c,0x21);
6861 SiS_SetReg(SiS_Part2Port,0x1d,0x45);
6862 SiS_SetReg(SiS_Part2Port,0x1f,0x0b);
6863 SiS_SetReg(SiS_Part2Port,0x20,0x00);
6864 SiS_SetReg(SiS_Part2Port,0x21,0xa9);
6865 SiS_SetReg(SiS_Part2Port,0x23,0x0b);
6866 SiS_SetReg(SiS_Part2Port,0x25,0x04);
6867 }
6868 }
6869 }
6870#endif
6871 }
6872}
6873
6874static void
6875SiS_SetGroup2(SiS_Private *SiS_Pr, USHORT ModeNo, USHORT ModeIdIndex,USHORT RefreshRateTableIndex,
6876 PSIS_HW_INFO HwInfo)
6877{
6878 USHORT i, j, tempax, tempbx, tempcx, tempch, tempcl, temp;
6879 USHORT push2, modeflag, crt2crtc, bridgeoffset;
6880 ULONG longtemp;
6881 const UCHAR *PhasePoint;
6882 const UCHAR *TimingPoint;
6883#ifdef SIS315H
6884 USHORT resindex, CRT2Index;
6885 const SiS_Part2PortTblStruct *CRT2Part2Ptr = NULL;
6886
6887 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA) return;
6888#endif
6889
6890 if(ModeNo <= 0x13) {
6891 modeflag = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_ModeFlag;
6892 crt2crtc = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_CRT2CRTC;
6893 } else if(SiS_Pr->UseCustomMode) {
6894 modeflag = SiS_Pr->CModeFlag;
6895 crt2crtc = 0;
6896 } else {
6897 modeflag = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_ModeFlag;
6898 crt2crtc = SiS_Pr->SiS_RefIndex[RefreshRateTableIndex].Ext_CRT2CRTC;
6899 }
6900
6901 temp = 0;
6902 if(!(SiS_Pr->SiS_VBInfo & SetCRT2ToAVIDEO)) temp |= 0x08;
6903 if(!(SiS_Pr->SiS_VBInfo & SetCRT2ToSVIDEO)) temp |= 0x04;
6904 if(SiS_Pr->SiS_VBInfo & SetCRT2ToSCART) temp |= 0x02;
6905 if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) temp |= 0x01;
6906
6907 if(!(SiS_Pr->SiS_TVMode & TVSetPAL)) temp |= 0x10;
6908
6909 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x00,temp);
6910
6911 PhasePoint = SiS_Pr->SiS_PALPhase;
6912 TimingPoint = SiS_Pr->SiS_PALTiming;
6913
6914 if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) {
6915
6916 TimingPoint = SiS_Pr->SiS_HiTVExtTiming;
6917 if(SiS_Pr->SiS_VBInfo & SetInSlaveMode) {
6918 TimingPoint = SiS_Pr->SiS_HiTVSt2Timing;
6919 if(SiS_Pr->SiS_TVMode & TVSetTVSimuMode) {
6920 TimingPoint = SiS_Pr->SiS_HiTVSt1Timing;
6921#if 0
6922 if(!(modeflag & Charx8Dot)) TimingPoint = SiS_Pr->SiS_HiTVTextTiming;
6923#endif
6924 }
6925 }
6926
6927 } else if(SiS_Pr->SiS_VBInfo & SetCRT2ToYPbPr525750) {
6928
6929 if(SiS_Pr->SiS_TVMode & TVSetYPbPr750p) TimingPoint = &SiS_YPbPrTable[2][0];
6930 else if(SiS_Pr->SiS_TVMode & TVSetYPbPr525p) TimingPoint = &SiS_YPbPrTable[1][0];
6931 else TimingPoint = &SiS_YPbPrTable[0][0];
6932
6933 PhasePoint = SiS_Pr->SiS_NTSCPhase;
6934
6935 } else if(SiS_Pr->SiS_TVMode & TVSetPAL) {
6936
6937 if( (SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) &&
6938 ( (!(SiS_Pr->SiS_VBInfo & SetInSlaveMode)) ||
6939 (SiS_Pr->SiS_TVMode & TVSetTVSimuMode) ) ) {
6940 PhasePoint = SiS_Pr->SiS_PALPhase2;
6941 }
6942
6943 } else {
6944
6945 TimingPoint = SiS_Pr->SiS_NTSCTiming;
6946 PhasePoint = SiS_Pr->SiS_NTSCPhase;
6947 if(SiS_Pr->SiS_TVMode & TVSetNTSCJ) {
6948 PhasePoint = SiS_Pr->SiS_PALPhase;
6949 }
6950
6951 if( (SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) &&
6952 ( (!(SiS_Pr->SiS_VBInfo & SetInSlaveMode)) ||
6953 (SiS_Pr->SiS_TVMode & TVSetTVSimuMode) ) ) {
6954 PhasePoint = SiS_Pr->SiS_NTSCPhase2;
6955 if(SiS_Pr->SiS_TVMode & TVSetNTSCJ) {
6956 PhasePoint = SiS_Pr->SiS_PALPhase2;
6957 }
6958 }
6959
6960 }
6961
6962 if(SiS_Pr->SiS_TVMode & TVSetPALM) {
6963 PhasePoint = SiS_Pr->SiS_PALMPhase;
6964 if( (SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) &&
6965 ( (!(SiS_Pr->SiS_VBInfo & SetInSlaveMode)) ||
6966 (SiS_Pr->SiS_TVMode & TVSetTVSimuMode) ) ) {
6967 PhasePoint = SiS_Pr->SiS_PALMPhase2;
6968 }
6969 }
6970
6971 if(SiS_Pr->SiS_TVMode & TVSetPALN) {
6972 PhasePoint = SiS_Pr->SiS_PALNPhase;
6973 if( (SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) &&
6974 ( (!(SiS_Pr->SiS_VBInfo & SetInSlaveMode)) ||
6975 (SiS_Pr->SiS_TVMode & TVSetTVSimuMode) ) ) {
6976 PhasePoint = SiS_Pr->SiS_PALNPhase2;
6977 }
6978 }
6979
6980 if(SiS_Pr->SiS_TVMode & TVSetNTSC1024) {
6981 PhasePoint = SiS_Pr->SiS_SpecialPhase;
6982 if(SiS_Pr->SiS_TVMode & TVSetPALM) {
6983 PhasePoint = SiS_Pr->SiS_SpecialPhaseM;
6984 } else if(SiS_Pr->SiS_TVMode & TVSetNTSCJ) {
6985 PhasePoint = SiS_Pr->SiS_SpecialPhaseJ;
6986 }
6987 }
6988
6989 for(i=0x31, j=0; i<=0x34; i++, j++) {
6990 SiS_SetReg(SiS_Pr->SiS_Part2Port,i,PhasePoint[j]);
6991 }
6992
6993 for(i=0x01, j=0; i<=0x2D; i++, j++) {
6994 SiS_SetReg(SiS_Pr->SiS_Part2Port,i,TimingPoint[j]);
6995 }
6996 for(i=0x39; i<=0x45; i++, j++) {
6997 SiS_SetReg(SiS_Pr->SiS_Part2Port,i,TimingPoint[j]);
6998 }
6999
7000 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) {
7001 if(SiS_Pr->SiS_ModeType != ModeText) {
7002 SiS_SetRegAND(SiS_Pr->SiS_Part2Port,0x3A,0x1F);
7003 }
7004 }
7005
7006 SiS_SetRegOR(SiS_Pr->SiS_Part2Port,0x0A,SiS_Pr->SiS_NewFlickerMode);
7007
7008 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x35,SiS_Pr->SiS_RY1COE);
7009 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x36,SiS_Pr->SiS_RY2COE);
7010 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x37,SiS_Pr->SiS_RY3COE);
7011 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x38,SiS_Pr->SiS_RY4COE);
7012
7013 if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) tempax = 950;
7014 else if(SiS_Pr->SiS_TVMode & TVSetPAL) tempax = 520;
7015 else tempax = 440; /* NTSC, YPbPr 525, 750 */
7016
7017 if( ( (!(SiS_Pr->SiS_VBInfo & SetCRT2ToTVNoHiVision)) && (SiS_Pr->SiS_VDE <= tempax) ) ||
7018 ( (SiS_Pr->SiS_VBInfo & SetCRT2ToTVNoHiVision) &&
7019 ((SiS_Pr->SiS_VGAHDE == 1024) || (SiS_Pr->SiS_VDE <= tempax)) ) ) {
7020
7021 tempax -= SiS_Pr->SiS_VDE;
7022 tempax >>= 2;
7023 tempax &= 0x00ff;
7024
7025 temp = tempax + (USHORT)TimingPoint[0];
7026 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x01,temp);
7027
7028 temp = tempax + (USHORT)TimingPoint[1];
7029 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x02,temp);
7030
7031 if((SiS_Pr->SiS_VBInfo & SetCRT2ToTVNoYPbPrHiVision) && (SiS_Pr->SiS_VGAHDE >= 1024)) {
7032 if(SiS_Pr->SiS_TVMode & TVSetPAL) {
7033 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x01,0x1b); /* 19 */
7034 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x02,0x54); /* 52 */
7035 } else {
7036 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x01,0x17);
7037 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x02,0x1d);
7038 }
7039 }
7040
7041 }
7042
7043 tempcx = SiS_Pr->SiS_HT;
7044 if(SiS_IsDualLink(SiS_Pr, HwInfo)) tempcx >>= 1;
7045 tempcx--;
7046 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) tempcx--;
7047 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x1B,tempcx);
7048 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x1D,0xF0,((tempcx >> 8) & 0x0f));
7049
7050 tempcx = SiS_Pr->SiS_HT >> 1;
7051 if(SiS_IsDualLink(SiS_Pr, HwInfo)) tempcx >>= 1;
7052 tempcx += 7;
7053 if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) tempcx -= 4;
7054 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x22,0x0F,((tempcx << 4) & 0xf0));
7055
7056 tempbx = TimingPoint[j] | (TimingPoint[j+1] << 8);
7057 tempbx += tempcx;
7058 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x24,tempbx);
7059 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x25,0x0F,((tempbx >> 4) & 0xf0));
7060
7061 tempbx += 8;
7062 if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) {
7063 tempbx -= 4;
7064 tempcx = tempbx;
7065 }
7066 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x29,0x0F,((tempbx << 4) & 0xf0));
7067
7068 j += 2;
7069 tempcx += (TimingPoint[j] | (TimingPoint[j+1] << 8));
7070 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x27,tempcx);
7071 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x28,0x0F,((tempcx >> 4) & 0xf0));
7072
7073 tempcx += 8;
7074 if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) tempcx -= 4;
7075 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x2A,0x0F,((tempcx << 4) & 0xf0));
7076
7077 tempcx = SiS_Pr->SiS_HT >> 1;
7078 if(SiS_IsDualLink(SiS_Pr, HwInfo)) tempcx >>= 1;
7079 j += 2;
7080 tempcx -= (TimingPoint[j] | ((TimingPoint[j+1]) << 8));
7081 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x2D,0x0F,((tempcx << 4) & 0xf0));
7082
7083 tempcx -= 11;
7084 if(!(SiS_Pr->SiS_VBInfo & SetCRT2ToTV)) {
7085 tempcx = SiS_GetVGAHT2(SiS_Pr) - 1;
7086 }
7087 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x2E,tempcx);
7088
7089 tempbx = SiS_Pr->SiS_VDE;
7090 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) {
7091 if(SiS_Pr->SiS_VGAVDE == 360) tempbx = 746;
7092 if(SiS_Pr->SiS_VGAVDE == 375) tempbx = 746;
7093 if(SiS_Pr->SiS_VGAVDE == 405) tempbx = 853;
7094 } else if( (SiS_Pr->SiS_VBInfo & SetCRT2ToTV) &&
7095 (!(SiS_Pr->SiS_TVMode & (TVSetYPbPr525p|TVSetYPbPr750p))) ) {
7096 tempbx >>= 1;
7097 if(HwInfo->jChipType >= SIS_315H) {
7098 if(SiS_Pr->SiS_TVMode & TVSetTVSimuMode) {
7099 if((ModeNo <= 0x13) && (crt2crtc == 1)) tempbx++;
7100 } else if(SiS_Pr->SiS_VBInfo & SetInSlaveMode) {
7101 if(SiS_Pr->SiS_ModeType <= ModeVGA) {
7102 if(crt2crtc == 4) tempbx++;
7103 }
7104 }
7105 }
7106 if(SiS_Pr->SiS_VBInfo & SetInSlaveMode) {
7107 if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) {
7108 if((ModeNo == 0x2f) || (ModeNo == 0x5d) || (ModeNo == 0x5e)) tempbx++;
7109 }
7110 if(!(SiS_Pr->SiS_TVMode & TVSetPAL)) {
7111 if(ModeNo == 0x03) tempbx++; /* From 1.10.7w - doesn't make sense */
7112 }
7113 }
7114 }
7115 tempbx -= 2;
7116 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x2F,tempbx);
7117
7118 temp = (tempcx >> 8) & 0x0F;
7119 temp |= ((tempbx >> 2) & 0xC0);
7120 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToSVIDEO | SetCRT2ToAVIDEO)) {
7121 temp |= 0x10;
7122 if(SiS_Pr->SiS_VBInfo & SetCRT2ToAVIDEO) temp |= 0x20;
7123 }
7124 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x30,temp);
7125
7126 if(SiS_Pr->SiS_VBType & (VB_SIS301C | VB_SIS302LV | VB_SIS302ELV)) {
7127 SiS_SetRegANDOR(SiS_Pr->SiS_Part4Port,0x10,0xdf,((tempbx & 0x0400) >> 5));
7128 }
7129
7130#if 0
7131 /* TEST qqqq */
7132 if(SiS_Pr->SiS_VBInfo & SetCRT2ToYPbPr525750) {
7133 for(i=0x01, j=0; i<=0x2D; i++, j++) {
7134 SiS_SetReg(SiS_Pr->SiS_Part2Port,i,TimingPoint[j]);
7135 }
7136 for(i=0x39; i<=0x45; i++, j++) {
7137 SiS_SetReg(SiS_Pr->SiS_Part2Port,i,TimingPoint[j]);
7138 }
7139 }
7140#endif
7141
7142 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) {
7143 tempbx = SiS_Pr->SiS_VDE;
7144 if( (SiS_Pr->SiS_VBInfo & SetCRT2ToTV) &&
7145 (!(SiS_Pr->SiS_TVMode & (TVSetYPbPr525p | TVSetYPbPr750p))) ) {
7146 tempbx >>= 1;
7147 }
7148 tempbx -= 3;
7149 temp = ((tempbx >> 3) & 0x60) | 0x18;
7150 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x46,temp);
7151 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x47,tempbx);
7152
7153 if(SiS_Pr->SiS_VBType & (VB_SIS301C | VB_SIS302LV | VB_SIS302ELV)) {
7154 SiS_SetRegANDOR(SiS_Pr->SiS_Part4Port,0x10,0xbf,((tempbx & 0x0400) >> 4));
7155 }
7156 }
7157
7158 tempbx = 0;
7159 if(!(modeflag & HalfDCLK)) {
7160 if(SiS_Pr->SiS_VGAHDE >= SiS_Pr->SiS_HDE) {
7161 tempax = 0;
7162 tempbx |= 0x20;
7163 }
7164 }
7165
7166 tempch = tempcl = 0x01;
7167 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) {
7168 if(SiS_Pr->SiS_VGAHDE >= 1024) {
7169 if((!(modeflag & HalfDCLK)) || (HwInfo->jChipType < SIS_315H)) {
7170 tempch = 0x19;
7171 tempcl = 0x20;
7172 if(SiS_Pr->SiS_VGAHDE >= 1280) {
7173 tempch = 0x14;
7174 tempbx &= ~0x20;
7175 }
7176 }
7177 }
7178 }
7179
7180 if(!(tempbx & 0x20)) {
7181 if(modeflag & HalfDCLK) tempcl <<= 1;
7182 longtemp = ((SiS_Pr->SiS_VGAHDE * tempch) / tempcl) << 13;
7183 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) longtemp <<= 3;
7184 tempax = longtemp / SiS_Pr->SiS_HDE;
7185 if(longtemp % SiS_Pr->SiS_HDE) tempax++;
7186 tempbx |= ((tempax >> 8) & 0x1F);
7187 tempcx = tempax >> 13;
7188 }
7189
7190 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x44,tempax);
7191 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x45,0xC0,tempbx);
7192
7193 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) {
7194
7195 tempcx &= 0x07;
7196 if(tempbx & 0x20) tempcx = 0;
7197 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x46,0xF8,tempcx);
7198
7199 if(SiS_Pr->SiS_TVMode & TVSetPAL) {
7200 tempbx = 0x0382;
7201 tempcx = 0x007e;
7202 } else {
7203 tempbx = 0x0369;
7204 tempcx = 0x0061;
7205 }
7206 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x4B,tempbx);
7207 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x4C,tempcx);
7208 temp = (tempcx & 0x0300) >> 6;
7209 temp |= ((tempbx >> 8) & 0x03);
7210 if(SiS_Pr->SiS_VBInfo & SetCRT2ToYPbPr525750) {
7211 temp |= 0x10;
7212 if(SiS_Pr->SiS_TVMode & TVSetYPbPr525p) temp |= 0x20;
7213 else if(SiS_Pr->SiS_TVMode & TVSetYPbPr750p) temp |= 0x40;
7214 }
7215 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x4D,temp);
7216
7217 temp = SiS_GetReg(SiS_Pr->SiS_Part2Port,0x43);
7218 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x43,(temp - 3));
7219
7220 SiS_SetTVSpecial(SiS_Pr, ModeNo);
7221
7222 if(SiS_Pr->SiS_VBType & VB_SIS301C) {
7223 temp = 0;
7224 if(SiS_Pr->SiS_TVMode & TVSetPALM) temp = 8;
7225 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x4e,0xf7,temp);
7226 }
7227
7228 }
7229
7230 if(SiS_Pr->SiS_TVMode & TVSetPALM) {
7231 if(!(SiS_Pr->SiS_TVMode & TVSetNTSC1024)) {
7232 temp = SiS_GetReg(SiS_Pr->SiS_Part2Port,0x01);
7233 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x01,(temp - 1));
7234 }
7235 SiS_SetRegAND(SiS_Pr->SiS_Part2Port,0x00,0xEF);
7236 }
7237
7238 if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) {
7239 if(!(SiS_Pr->SiS_VBInfo & SetInSlaveMode)) {
7240 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x0B,0x00);
7241 }
7242 }
7243
7244 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) return;
7245
7246 /* From here: Part2 LCD setup */
7247
7248 tempbx = SiS_Pr->SiS_HDE;
7249 if(SiS_IsDualLink(SiS_Pr, HwInfo)) tempbx >>= 1;
7250 tempbx--; /* RHACTE = HDE - 1 */
7251 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x2C,tempbx);
7252 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x2B,0x0F,((tempbx >> 4) & 0xf0));
7253
7254 temp = 0x01;
7255 if(SiS_Pr->SiS_LCDResInfo == Panel_1280x1024) {
7256 if(SiS_Pr->SiS_ModeType == ModeEGA) {
7257 if(SiS_Pr->SiS_VGAHDE >= 1024) {
7258 temp = 0x02;
7259 if(HwInfo->jChipType >= SIS_315H) {
7260 if(SiS_Pr->SiS_SetFlag & LCDVESATiming) {
7261 temp = 0x01;
7262 }
7263 }
7264 }
7265 }
7266 }
7267 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x0B,temp);
7268
7269 tempbx = SiS_Pr->SiS_VDE - 1;
7270 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x03,tempbx);
7271 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x0C,0xF8,((tempbx >> 8) & 0x07));
7272
7273 tempcx = SiS_Pr->SiS_VT - 1;
7274 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x19,tempcx);
7275 temp = (tempcx >> 3) & 0xE0;
7276 if(SiS_Pr->SiS_LCDInfo & LCDRGB18Bit) {
7277 /* Enable dithering; only do this for 32bpp mode */
7278 if(SiS_GetReg(SiS_Pr->SiS_Part1Port,0x00) & 0x01) {
7279 temp |= 0x10;
7280 }
7281 }
7282 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x1A,0x0f,temp);
7283
7284 SiS_SetRegAND(SiS_Pr->SiS_Part2Port,0x09,0xF0);
7285 SiS_SetRegAND(SiS_Pr->SiS_Part2Port,0x0A,0xF0);
7286
7287 SiS_SetRegAND(SiS_Pr->SiS_Part2Port,0x17,0xFB);
7288 SiS_SetRegAND(SiS_Pr->SiS_Part2Port,0x18,0xDF);
7289
7290#ifdef SIS315H
7291 if(SiS_GetCRT2Part2Ptr(SiS_Pr, ModeNo, ModeIdIndex, RefreshRateTableIndex,
7292 &CRT2Index, &resindex, HwInfo)) {
7293 switch(CRT2Index) {
7294 case 200: CRT2Part2Ptr = SiS_Pr->SiS_CRT2Part2_1024x768_1; break;
7295 case 206: CRT2Part2Ptr = SiS310_CRT2Part2_Asus1024x768_3; break;
7296 default: CRT2Part2Ptr = SiS_Pr->SiS_CRT2Part2_1024x768_3; break;
7297 }
7298
7299 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x01,0x80,(CRT2Part2Ptr+resindex)->CR[0]);
7300 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x02,0x80,(CRT2Part2Ptr+resindex)->CR[1]);
7301 for(i = 2, j = 0x04; j <= 0x06; i++, j++ ) {
7302 SiS_SetReg(SiS_Pr->SiS_Part2Port,j,(CRT2Part2Ptr+resindex)->CR[i]);
7303 }
7304 for(j = 0x1c; j <= 0x1d; i++, j++ ) {
7305 SiS_SetReg(SiS_Pr->SiS_Part2Port,j,(CRT2Part2Ptr+resindex)->CR[i]);
7306 }
7307 for(j = 0x1f; j <= 0x21; i++, j++ ) {
7308 SiS_SetReg(SiS_Pr->SiS_Part2Port,j,(CRT2Part2Ptr+resindex)->CR[i]);
7309 }
7310 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x23,(CRT2Part2Ptr+resindex)->CR[10]);
7311 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x25,0x0f,(CRT2Part2Ptr+resindex)->CR[11]);
7312
7313 SiS_SetGroup2_Tail(SiS_Pr, ModeNo);
7314
7315
7316 } else {
7317#endif
7318
7319 /* Checked for 1024x768, 1280x1024, 1400x1050, 1600x1200 */
7320 /* Clevo dual-link 1024x768 */
7321 /* Compaq 1280x1024 has HT 1696 sometimes (calculation OK, if given HT is correct) */
7322 /* Acer: OK, but uses different setting for VESA timing at 640/800/1024 and 640x400 */
7323
7324 if(SiS_Pr->SiS_LCDInfo & DontExpandLCD) {
7325 if((SiS_Pr->SiS_LCDInfo & LCDPass11) || (SiS_Pr->PanelYRes == SiS_Pr->SiS_VDE)) {
7326 tempbx = SiS_Pr->SiS_VDE - 1;
7327 tempcx = SiS_Pr->SiS_VT - 1;
7328 } else {
7329 tempbx = SiS_Pr->SiS_VDE + ((SiS_Pr->PanelYRes - SiS_Pr->SiS_VDE) / 2);
7330 tempcx = SiS_Pr->SiS_VT - ((SiS_Pr->PanelYRes - SiS_Pr->SiS_VDE) / 2);
7331 }
7332 } else {
7333 tempbx = SiS_Pr->PanelYRes;
7334 tempcx = SiS_Pr->SiS_VT;
7335 tempax = 1;
7336 if(SiS_Pr->PanelYRes != SiS_Pr->SiS_VDE) {
7337 tempax = SiS_Pr->PanelYRes;
7338 /* if(SiS_Pr->SiS_VGAVDE == 525) tempax += 0x3c; */ /* 651+301C */
7339 if(SiS_Pr->PanelYRes < SiS_Pr->SiS_VDE) {
7340 tempax = tempcx = 0;
7341 } else {
7342 tempax -= SiS_Pr->SiS_VDE;
7343 }
7344 tempax >>= 1;
7345 }
7346 tempcx -= tempax; /* lcdvdes */
7347 tempbx -= tempax; /* lcdvdee */
7348 }
7349
7350 /* Non-expanding: lcdvdes = tempcx = VT-1; lcdvdee = tempbx = VDE-1 */
7351
7352#ifdef TWDEBUG
7353 xf86DrvMsg(0, X_INFO, "lcdvdes 0x%x lcdvdee 0x%x\n", tempcx, tempbx);
7354#endif
7355
7356 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x05,tempcx); /* lcdvdes */
7357 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x06,tempbx); /* lcdvdee */
7358
7359 temp = (tempbx >> 5) & 0x38;
7360 temp |= ((tempcx >> 8) & 0x07);
7361 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x02,temp);
7362
7363 tempax = SiS_Pr->SiS_VDE;
7364 if((SiS_Pr->SiS_LCDInfo & DontExpandLCD) && (!(SiS_Pr->SiS_LCDInfo & LCDPass11))) {
7365 tempax = SiS_Pr->PanelYRes;
7366 }
7367 tempcx = (SiS_Pr->SiS_VT - tempax) >> 4;
7368 if((SiS_Pr->SiS_LCDInfo & DontExpandLCD) && (!(SiS_Pr->SiS_LCDInfo & LCDPass11))) {
7369 if(SiS_Pr->PanelYRes != SiS_Pr->SiS_VDE) {
7370 tempcx = (SiS_Pr->SiS_VT - tempax) / 10;
7371 }
7372 }
7373
7374 tempbx = ((SiS_Pr->SiS_VT + SiS_Pr->SiS_VDE) >> 1) - 1;
7375 if(SiS_Pr->SiS_LCDInfo & DontExpandLCD) {
7376 if(SiS_Pr->PanelYRes != SiS_Pr->SiS_VDE) {
7377 if(!(SiS_Pr->SiS_LCDInfo & LCDPass11)) { /* ? */
7378 tempax = SiS_Pr->SiS_VT - SiS_Pr->PanelYRes;
7379 if(tempax % 4) { tempax >>= 2; tempax++; }
7380 else { tempax >>= 2; }
7381 tempbx -= (tempax - 1);
7382 } else {
7383 tempbx -= 10;
7384 if(tempbx <= SiS_Pr->SiS_VDE) tempbx = SiS_Pr->SiS_VDE + 1;
7385 }
7386 }
7387 }
7388 if(SiS_Pr->SiS_LCDResInfo == Panel_1024x768) {
7389 tempbx++;
7390 if((!(SiS_Pr->SiS_LCDInfo & DontExpandLCD)) || (crt2crtc == 6)) {
7391 if(SiS_Pr->SiS_SetFlag & LCDVESATiming) {
7392 tempbx = 770;
7393 tempcx = 3;
7394 }
7395 }
7396 }
7397
7398 /* non-expanding: lcdvrs = ((VT + VDE) / 2) - 10 */
7399
7400 if(SiS_Pr->UseCustomMode) {
7401 tempbx = SiS_Pr->CVSyncStart;
7402 }
7403
7404#ifdef TWDEBUG
7405 xf86DrvMsg(0, X_INFO, "lcdvrs 0x%x\n", tempbx);
7406#endif
7407
7408 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x04,tempbx); /* lcdvrs */
7409
7410 temp = (tempbx >> 4) & 0xF0;
7411 tempbx += (tempcx + 1);
7412 temp |= (tempbx & 0x0F);
7413
7414 if(SiS_Pr->UseCustomMode) {
7415 temp &= 0xf0;
7416 temp |= (SiS_Pr->CVSyncEnd & 0x0f);
7417 }
7418
7419#ifdef TWDEBUG
7420 xf86DrvMsg(0, X_INFO, "lcdvre[3:0] 0x%x\n", (temp & 0x0f));
7421#endif
7422
7423 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x01,temp);
7424
7425#ifdef SIS300
7426 SiS_Group2LCDSpecial(SiS_Pr, HwInfo, ModeNo, crt2crtc);
7427#endif
7428
7429 bridgeoffset = 7;
7430 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) bridgeoffset += 2;
7431 if(SiS_Pr->SiS_VBType & (VB_SIS301C | VB_SIS302ELV)) bridgeoffset++;
7432 if(SiS_IsDualLink(SiS_Pr, HwInfo)) bridgeoffset++;
7433
7434 temp = 0;
7435 if((SiS_Pr->SiS_LCDInfo & DontExpandLCD) && (!(SiS_Pr->SiS_LCDInfo & LCDPass11))) {
7436 if(SiS_Pr->PanelXRes != SiS_Pr->SiS_HDE) {
7437 temp = SiS_Pr->SiS_HT - ((SiS_Pr->PanelXRes - SiS_Pr->SiS_HDE) / 2);
7438 if(SiS_IsDualLink(SiS_Pr, HwInfo)) temp >>= 1;
7439 }
7440 }
7441 temp += bridgeoffset;
7442 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x1F,temp); /* lcdhdes */
7443 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x20,0x0F,((temp >> 4) & 0xf0));
7444
7445 tempcx = SiS_Pr->SiS_HT;
7446 tempax = tempbx = SiS_Pr->SiS_HDE;
7447 if((SiS_Pr->SiS_LCDInfo & DontExpandLCD) && (!(SiS_Pr->SiS_LCDInfo & LCDPass11))) {
7448 if(SiS_Pr->PanelXRes != SiS_Pr->SiS_HDE) {
7449 tempax = SiS_Pr->PanelXRes;
7450 tempbx = SiS_Pr->PanelXRes - ((SiS_Pr->PanelXRes - SiS_Pr->SiS_HDE) / 2);
7451 }
7452 }
7453 if(SiS_IsDualLink(SiS_Pr, HwInfo)) {
7454 tempcx >>= 1;
7455 tempbx >>= 1;
7456 tempax >>= 1;
7457 }
7458
7459#ifdef TWDEBUG
7460 xf86DrvMsg(0, X_INFO, "lcdhdee 0x%x\n", tempbx);
7461#endif
7462
7463 tempbx += bridgeoffset;
7464
7465 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x23,tempbx); /* lcdhdee */
7466 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x25,0xF0,((tempbx >> 8) & 0x0f));
7467
7468 tempcx = (tempcx - tempax) >> 2;
7469
7470 tempbx += tempcx;
7471 push2 = tempbx;
7472
7473 if(SiS_Pr->SiS_LCDResInfo == Panel_1280x1024) {
7474 if(SiS_Pr->SiS_LCDInfo & DontExpandLCD) {
7475 if(SiS_Pr->SiS_LCDInfo & LCDPass11) {
7476 if(SiS_Pr->SiS_HDE == 1280) tempbx = (tempbx & 0xff00) | 0x47;
7477 }
7478 }
7479 }
7480
7481 if(SiS_Pr->UseCustomMode) {
7482 tempbx = SiS_Pr->CHSyncStart;
7483 if(SiS_IsDualLink(SiS_Pr, HwInfo)) tempbx >>= 1;
7484 tempbx += bridgeoffset;
7485 }
7486
7487#ifdef TWDEBUG
7488 xf86DrvMsg(0, X_INFO, "lcdhrs 0x%x\n", tempbx);
7489#endif
7490
7491 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x1C,tempbx); /* lcdhrs */
7492 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x1D,0x0F,((tempbx >> 4) & 0xf0));
7493
7494 tempbx = push2;
7495
7496 tempcx <<= 1;
7497 if((SiS_Pr->SiS_LCDInfo & DontExpandLCD) && (!(SiS_Pr->SiS_LCDInfo & LCDPass11))) {
7498 if(SiS_Pr->PanelXRes != SiS_Pr->SiS_HDE) tempcx >>= 2;
7499 }
7500 tempbx += tempcx;
7501
7502 if(SiS_Pr->UseCustomMode) {
7503 tempbx = SiS_Pr->CHSyncEnd;
7504 if(SiS_IsDualLink(SiS_Pr, HwInfo)) tempbx >>= 1;
7505 tempbx += bridgeoffset;
7506 }
7507
7508#ifdef TWDEBUG
7509 xf86DrvMsg(0, X_INFO, "lcdhre 0x%x\n", tempbx);
7510#endif
7511
7512 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x21,tempbx); /* lcdhre */
7513
7514 SiS_SetGroup2_Tail(SiS_Pr, ModeNo);
7515
7516#ifdef SIS300
7517 SiS_Set300Part2Regs(SiS_Pr, HwInfo, ModeIdIndex, RefreshRateTableIndex, ModeNo);
7518#endif
7519#ifdef SIS315H
7520 } /* CRT2-LCD from table */
7521#endif
7522}
7523
7524/*********************************************/
7525/* SET PART 3 REGISTER GROUP */
7526/*********************************************/
7527
7528static void
7529SiS_SetGroup3(SiS_Private *SiS_Pr, USHORT ModeNo, USHORT ModeIdIndex,
7530 PSIS_HW_INFO HwInfo)
7531{
7532 USHORT i;
7533 const UCHAR *tempdi;
7534
7535 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA) return;
7536
7537#ifndef SIS_CP
7538 SiS_SetReg(SiS_Pr->SiS_Part3Port,0x00,0x00);
7539#else
7540 SIS_CP_INIT301_CP
7541#endif
7542
7543 if(SiS_Pr->SiS_TVMode & TVSetPAL) {
7544 SiS_SetReg(SiS_Pr->SiS_Part3Port,0x13,0xFA);
7545 SiS_SetReg(SiS_Pr->SiS_Part3Port,0x14,0xC8);
7546 } else {
7547 SiS_SetReg(SiS_Pr->SiS_Part3Port,0x13,0xF5);
7548 SiS_SetReg(SiS_Pr->SiS_Part3Port,0x14,0xB7);
7549 }
7550
7551 if(SiS_Pr->SiS_TVMode & TVSetPALM) {
7552 SiS_SetReg(SiS_Pr->SiS_Part3Port,0x13,0xFA);
7553 SiS_SetReg(SiS_Pr->SiS_Part3Port,0x14,0xC8);
7554 SiS_SetReg(SiS_Pr->SiS_Part3Port,0x3D,0xA8);
7555 }
7556
7557 tempdi = NULL;
7558 if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) {
7559 tempdi = SiS_Pr->SiS_HiTVGroup3Data;
7560 if(SiS_Pr->SiS_TVMode & TVSetTVSimuMode) {
7561 tempdi = SiS_Pr->SiS_HiTVGroup3Simu;
7562 }
7563 } else if(SiS_Pr->SiS_VBInfo & SetCRT2ToYPbPr525750) {
7564 if(!(SiS_Pr->SiS_TVMode & TVSetYPbPr525i)) {
7565 tempdi = SiS_HiTVGroup3_1;
7566 if(SiS_Pr->SiS_TVMode & TVSetYPbPr750p) tempdi = SiS_HiTVGroup3_2;
7567 }
7568 }
7569 if(tempdi) {
7570 for(i=0; i<=0x3E; i++) {
7571 SiS_SetReg(SiS_Pr->SiS_Part3Port,i,tempdi[i]);
7572 }
7573 if(SiS_Pr->SiS_VBType & (VB_SIS301C | VB_SIS302ELV)) {
7574 if(SiS_Pr->SiS_TVMode & TVSetYPbPr525p) {
7575 SiS_SetReg(SiS_Pr->SiS_Part3Port,0x28,0x3f);
7576 }
7577 }
7578 }
7579
7580#ifdef SIS_CP
7581 SIS_CP_INIT301_CP2
7582#endif
7583}
7584
7585/*********************************************/
7586/* SET PART 4 REGISTER GROUP */
7587/*********************************************/
7588
7589#ifdef SIS315H
7590static void
7591SiS_ShiftXPos(SiS_Private *SiS_Pr, int shift)
7592{
7593 USHORT temp, temp1, temp2;
7594
7595 temp1 = SiS_GetReg(SiS_Pr->SiS_Part2Port,0x1f);
7596 temp2 = SiS_GetReg(SiS_Pr->SiS_Part2Port,0x20);
7597 temp = (USHORT)((int)((temp1 | ((temp2 & 0xf0) << 4))) + shift);
7598 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x1f,temp);
7599 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x20,0x0f,((temp >> 4) & 0xf0));
7600 temp = SiS_GetReg(SiS_Pr->SiS_Part2Port,0x2b) & 0x0f;
7601 temp = (USHORT)((int)(temp) + shift);
7602 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x2b,0xf0,(temp & 0x0f));
7603 temp1 = SiS_GetReg(SiS_Pr->SiS_Part2Port,0x43);
7604 temp2 = SiS_GetReg(SiS_Pr->SiS_Part2Port,0x42);
7605 temp = (USHORT)((int)((temp1 | ((temp2 & 0xf0) << 4))) + shift);
7606 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x43,temp);
7607 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x42,0x0f,((temp >> 4) & 0xf0));
7608}
7609
7610static void
7611SiS_SetGroup4_C_ELV(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo,
7612 USHORT ModeNo, USHORT ModeIdIndex)
7613{
7614 USHORT temp, temp1, resinfo = 0;
7615
7616 if(!(SiS_Pr->SiS_VBType & VB_SIS301C)) return;
7617 if(!(SiS_Pr->SiS_VBInfo & (SetCRT2ToHiVision | SetCRT2ToYPbPr525750))) return;
7618
7619 if(ModeNo > 0x13) {
7620 resinfo = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_RESINFO;
7621 }
7622
7623 SiS_SetRegOR(SiS_Pr->SiS_Part4Port,0x3a,0x08);
7624 temp = SiS_GetReg(SiS_Pr->SiS_Part4Port,0x3a);
7625 if(!(temp & 0x01)) {
7626 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x3a,0xdf);
7627 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x25,0xfc);
7628 if((HwInfo->jChipType < SIS_661) && (!(SiS_Pr->SiS_ROMNew))) {
7629 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x25,0xf8);
7630 }
7631 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x0f,0xfb);
7632 if(SiS_Pr->SiS_TVMode & TVSetYPbPr750p) temp = 0x0000;
7633 else if(SiS_Pr->SiS_TVMode & TVSetYPbPr525p) temp = 0x0002;
7634 else if(SiS_Pr->SiS_TVMode & TVSetHiVision) temp = 0x0400;
7635 else temp = 0x0402;
7636 if((HwInfo->jChipType >= SIS_661) || (SiS_Pr->SiS_ROMNew)) {
7637 temp1 = 0;
7638 if(SiS_Pr->SiS_TVMode & TVAspect43) temp1 = 4;
7639 SiS_SetRegANDOR(SiS_Pr->SiS_Part4Port,0x0f,0xfb,temp1);
7640 if(SiS_Pr->SiS_TVMode & TVAspect43LB) temp |= 0x01;
7641 SiS_SetRegANDOR(SiS_Pr->SiS_Part4Port,0x26,0x7c,(temp & 0xff));
7642 } else {
7643 temp1 = SiS_GetReg(SiS_Pr->SiS_P3d4,0x3b) & 0x03;
7644 if(temp1 == 0x01) temp |= 0x01;
7645 if(temp1 == 0x03) temp |= 0x04; /* ? why not 0x10? */
7646 SiS_SetRegANDOR(SiS_Pr->SiS_Part4Port,0x26,0xf8,(temp & 0xff));
7647 }
7648 SiS_SetRegANDOR(SiS_Pr->SiS_Part4Port,0x3a,0xfb,(temp >> 8));
7649 if(ModeNo > 0x13) {
7650 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x3b,0xfd);
7651 }
7652
7653 if(HwInfo->jChipType >= SIS_661) { /* ? */
7654 if(SiS_Pr->SiS_TVMode & TVAspect43) {
7655 if(SiS_Pr->SiS_TVMode & TVSetYPbPr750p) {
7656 if(resinfo == SIS_RI_1024x768) {
7657 SiS_ShiftXPos(SiS_Pr, 97);
7658 } else {
7659 SiS_ShiftXPos(SiS_Pr, 111);
7660 }
7661 } else if(SiS_Pr->SiS_TVMode & TVSetHiVision) {
7662 SiS_ShiftXPos(SiS_Pr, 136);
7663 }
7664 }
7665 }
7666 }
7667}
7668#endif
7669
7670static void
7671SiS_SetCRT2VCLK(SiS_Private *SiS_Pr, USHORT ModeNo, USHORT ModeIdIndex,
7672 USHORT RefreshRateTableIndex, PSIS_HW_INFO HwInfo)
7673{
7674 USHORT vclkindex;
7675 USHORT temp, reg1, reg2;
7676
7677 if(SiS_Pr->UseCustomMode) {
7678 reg1 = SiS_Pr->CSR2B;
7679 reg2 = SiS_Pr->CSR2C;
7680 } else {
7681 vclkindex = SiS_GetVCLK2Ptr(SiS_Pr, ModeNo, ModeIdIndex, RefreshRateTableIndex,
7682 HwInfo);
7683 reg1 = SiS_Pr->SiS_VBVCLKData[vclkindex].Part4_A;
7684 reg2 = SiS_Pr->SiS_VBVCLKData[vclkindex].Part4_B;
7685 }
7686
7687 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) {
7688 if(SiS_Pr->SiS_TVMode & TVSetNTSC1024) {
7689 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x0a,0x57);
7690 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x0b,0x46);
7691 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x1f,0xf6);
7692 } else {
7693 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x0a,reg1);
7694 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x0b,reg2);
7695 }
7696 } else {
7697 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x0a,0x01);
7698 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x0b,reg2);
7699 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x0a,reg1);
7700 }
7701 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x12,0x00);
7702 temp = 0x08;
7703 if(SiS_Pr->SiS_VBInfo & SetCRT2ToRAMDAC) temp |= 0x20;
7704 SiS_SetRegOR(SiS_Pr->SiS_Part4Port,0x12,temp);
7705}
7706
7707static void
7708SiS_SetGroup4(SiS_Private *SiS_Pr, USHORT ModeNo, USHORT ModeIdIndex,
7709 USHORT RefreshRateTableIndex, PSIS_HW_INFO HwInfo)
7710{
7711 USHORT tempax,tempcx,tempbx,modeflag,temp,resinfo;
7712 ULONG tempebx,tempeax,templong;
7713
7714 if(ModeNo <= 0x13) {
7715 modeflag = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_ModeFlag;
7716 resinfo = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_ResInfo;
7717 } else if(SiS_Pr->UseCustomMode) {
7718 modeflag = SiS_Pr->CModeFlag;
7719 resinfo = 0;
7720 } else {
7721 modeflag = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_ModeFlag;
7722 resinfo = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_RESINFO;
7723 }
7724
7725 if(HwInfo->jChipType >= SIS_315H) {
7726 if(SiS_Pr->SiS_VBType & VB_SIS301LV302LV) {
7727 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA) {
7728 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x24,0x0e);
7729 }
7730 }
7731 }
7732
7733 if(SiS_Pr->SiS_VBType & (VB_SIS301C | VB_SIS302LV)) {
7734 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) {
7735 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x10,0x9f);
7736 }
7737 }
7738
7739 if(HwInfo->jChipType >= SIS_315H) {
7740 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA) {
7741 if(SiS_Pr->SiS_VBType & VB_SIS301LV302LV) {
7742 if(SiS_IsDualLink(SiS_Pr, HwInfo)) {
7743 SiS_SetRegOR(SiS_Pr->SiS_Part4Port,0x27,0x2c);
7744 } else {
7745 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x27,~0x20);
7746 }
7747
7748 if(SiS_Pr->SiS_VBType & (VB_SIS302LV | VB_SIS302ELV)) {
7749 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x2a,0x00);
7750#ifdef SET_EMI
7751 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x30,0x0c);
7752#endif
7753 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x34,0x10);
7754 }
7755 }
7756 return;
7757 }
7758 }
7759
7760 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x13,SiS_Pr->SiS_RVBHCFACT);
7761
7762 tempbx = SiS_Pr->SiS_RVBHCMAX;
7763 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x14,tempbx);
7764
7765 temp = (tempbx >> 1) & 0x80;
7766
7767 tempcx = SiS_Pr->SiS_VGAHT - 1;
7768 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x16,tempcx);
7769
7770 temp |= ((tempcx >> 5) & 0x78);
7771
7772 tempcx = SiS_Pr->SiS_VGAVT - 1;
7773 if(!(SiS_Pr->SiS_VBInfo & SetCRT2ToTV)) tempcx -= 5;
7774 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x17,tempcx);
7775
7776 temp |= ((tempcx >> 8) & 0x07);
7777 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x15,temp);
7778
7779 tempbx = SiS_Pr->SiS_VGAHDE;
7780 if(modeflag & HalfDCLK) tempbx >>= 1;
7781 if(SiS_IsDualLink(SiS_Pr, HwInfo)) tempbx >>= 1;
7782
7783 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) {
7784 temp = 0;
7785 if(tempbx > 800) temp = 0x60;
7786 } else if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) {
7787 temp = 0;
7788 if(tempbx == 1024) temp = 0xA0;
7789 else if(tempbx > 1024) temp = 0xC0;
7790 } else if(SiS_Pr->SiS_TVMode & (TVSetYPbPr525p | TVSetYPbPr750p)) {
7791 temp = 0;
7792 if(tempbx >= 1280) temp = 0x40;
7793 else if(tempbx >= 1024) temp = 0x20;
7794 } else {
7795 temp = 0x80;
7796 if(tempbx >= 1024) temp = 0xA0;
7797 }
7798
7799 if(SiS_Pr->SiS_VBType & VB_SIS301) {
7800 if(SiS_Pr->SiS_LCDResInfo != Panel_1280x1024) temp |= 0x0A;
7801 }
7802
7803 SiS_SetRegANDOR(SiS_Pr->SiS_Part4Port,0x0E,0x10,temp);
7804
7805 tempeax = SiS_Pr->SiS_VGAVDE;
7806 tempebx = SiS_Pr->SiS_VDE;
7807 if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) {
7808 if(!(temp & 0xE0)) tempebx >>=1;
7809 }
7810
7811 tempcx = SiS_Pr->SiS_RVBHRS;
7812 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x18,tempcx);
7813 tempcx >>= 8;
7814 tempcx |= 0x40;
7815
7816 if(tempeax <= tempebx) {
7817 tempcx ^= 0x40;
7818 } else {
7819 tempeax -= tempebx;
7820 }
7821
7822 tempeax *= (256 * 1024);
7823 templong = tempeax % tempebx;
7824 tempeax /= tempebx;
7825 if(templong) tempeax++;
7826
7827 temp = (USHORT)(tempeax & 0x000000FF);
7828 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x1B,temp);
7829 temp = (USHORT)((tempeax & 0x0000FF00) >> 8);
7830 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x1A,temp);
7831 temp = (USHORT)((tempeax >> 12) & 0x70); /* sic! */
7832 temp |= (tempcx & 0x4F);
7833 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x19,temp);
7834
7835 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) {
7836
7837 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x1C,0x28);
7838
7839 /* Calc Linebuffer max address and set/clear decimode */
7840 tempbx = 0;
7841 if(SiS_Pr->SiS_TVMode & (TVSetHiVision | TVSetYPbPr750p)) tempbx = 0x08;
7842 tempax = SiS_Pr->SiS_VGAHDE;
7843 if(modeflag & HalfDCLK) tempax >>= 1;
7844 if(SiS_IsDualLink(SiS_Pr, HwInfo)) tempax >>= 1;
7845 if(tempax > 800) {
7846 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) {
7847 tempax -= 800;
7848 } else { /* 651+301C: Only if TVNoHiviNoYPbPr */
7849 tempbx = 0x08;
7850 if(tempax == 1024) tempax *= 25;
7851 else tempax *= 20;
7852 temp = tempax % 32;
7853 tempax /= 32;
7854 if(temp) tempax++;
7855 tempax++;
7856 if((SiS_Pr->SiS_VBInfo & SetCRT2ToTVNoYPbPrHiVision) ||
7857 (SiS_Pr->SiS_TVMode & TVSetYPbPr525i)) {
7858 if(resinfo == SIS_RI_1024x768) {
7859 /* Otherwise white line at right edge */
7860 tempax = (tempax & 0xff00) | 0x20;
7861 }
7862 }
7863 }
7864 }
7865 tempax--;
7866 temp = ((tempax >> 4) & 0x30) | tempbx;
7867 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x1D,tempax);
7868 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x1E,temp);
7869
7870 temp = 0x0036; tempbx = 0xD0;
7871 if((HwInfo->jChipType >= SIS_315H) && (SiS_Pr->SiS_VBType & VB_SIS301LV302LV)) {
7872 temp = 0x0026; tempbx = 0xC0; /* See En/DisableBridge() */
7873 }
7874 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) {
7875 if(!(SiS_Pr->SiS_TVMode & (TVSetNTSC1024 | TVSetHiVision | TVSetYPbPr750p | TVSetYPbPr525p))) {
7876 temp |= 0x01;
7877 if(SiS_Pr->SiS_VBInfo & SetInSlaveMode) {
7878 if(!(SiS_Pr->SiS_TVMode & TVSetTVSimuMode)) {
7879 temp &= ~0x01;
7880 }
7881 }
7882 }
7883 }
7884 SiS_SetRegANDOR(SiS_Pr->SiS_Part4Port,0x1F,tempbx,temp);
7885
7886 tempbx = SiS_Pr->SiS_HT >> 1;
7887 if(SiS_IsDualLink(SiS_Pr, HwInfo)) tempbx >>= 1;
7888 tempbx -= 2;
7889 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x22,tempbx);
7890 temp = (tempbx >> 5) & 0x38;
7891 SiS_SetRegANDOR(SiS_Pr->SiS_Part4Port,0x21,0xC0,temp);
7892
7893 if(SiS_Pr->SiS_VBType & VB_SIS301LV302LV) {
7894 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) {
7895 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x24,0x0e);
7896 /* LCD-too-dark-error-source, see FinalizeLCD() */
7897 }
7898 if(HwInfo->jChipType >= SIS_315H) {
7899 if(SiS_IsDualLink(SiS_Pr, HwInfo)) {
7900 SiS_SetRegOR(SiS_Pr->SiS_Part4Port,0x27,0x2c);
7901 } else {
7902 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x27,~0x20);
7903 }
7904 }
7905 if(SiS_Pr->SiS_VBType & (VB_SIS302LV | VB_SIS302ELV)) {
7906 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x2a,0x00);
7907#ifdef SET_EMI
7908 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x30,0x0c);
7909#endif
7910 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x34,0x10);
7911 }
7912 }
7913
7914 } /* 301B */
7915
7916 SiS_SetCRT2VCLK(SiS_Pr, ModeNo, ModeIdIndex, RefreshRateTableIndex, HwInfo);
7917}
7918
7919/*********************************************/
7920/* SET PART 5 REGISTER GROUP */
7921/*********************************************/
7922
7923static void
7924SiS_SetGroup5(SiS_Private *SiS_Pr, USHORT ModeNo, USHORT ModeIdIndex,
7925 PSIS_HW_INFO HwInfo)
7926{
7927
7928 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA) return;
7929
7930 if(SiS_Pr->SiS_ModeType == ModeVGA) {
7931 if(!(SiS_Pr->SiS_VBInfo & (SetInSlaveMode | LoadDACFlag))) {
7932 SiS_SetRegOR(SiS_Pr->SiS_P3c4,0x1E,0x20);
7933 SiS_LoadDAC(SiS_Pr, HwInfo, ModeNo, ModeIdIndex);
7934 }
7935 }
7936}
7937
7938/*********************************************/
7939/* MODIFY CRT1 GROUP FOR SLAVE MODE */
7940/*********************************************/
7941
7942static void
7943SiS_ModCRT1CRTC(SiS_Private *SiS_Pr, USHORT ModeNo, USHORT ModeIdIndex,
7944 USHORT RefreshRateTableIndex, PSIS_HW_INFO HwInfo)
7945{
7946 USHORT tempah,i,modeflag,j;
7947 USHORT ResIndex,DisplayType;
7948 const SiS_LVDSCRT1DataStruct *LVDSCRT1Ptr=NULL;
7949
7950 if(ModeNo <= 0x13) modeflag = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_ModeFlag;
7951 else modeflag = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_ModeFlag;
7952
7953 if((SiS_Pr->SiS_CustomT == CUT_BARCO1366) ||
7954 (SiS_Pr->SiS_CustomT == CUT_BARCO1024) ||
7955 (SiS_Pr->SiS_CustomT == CUT_PANEL848))
7956 return;
7957
7958 if(!(SiS_GetLVDSCRT1Ptr(SiS_Pr, ModeNo, ModeIdIndex, RefreshRateTableIndex,
7959 &ResIndex, &DisplayType))) {
7960 return;
7961 }
7962
7963 if(HwInfo->jChipType < SIS_315H) {
7964 if(SiS_Pr->SiS_SetFlag & SetDOSMode) return;
7965 }
7966
7967 switch(DisplayType) {
7968 case 0 : LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT1800x600_1; break;
7969 case 1 : LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT1800x600_1_H; break;
7970 case 2 : LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT1800x600_2; break;
7971 case 3 : LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT1800x600_2_H; break;
7972 case 4 : LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11024x768_1; break;
7973 case 5 : LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11024x768_1_H; break;
7974 case 6 : LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11024x768_2; break;
7975 case 7 : LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11024x768_2_H; break;
7976 case 8 : LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11280x1024_1; break;
7977 case 9 : LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11280x1024_1_H; break;
7978 case 10: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11280x1024_2; break;
7979 case 11: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11280x1024_2_H; break;
7980 case 12: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT1XXXxXXX_1; break;
7981 case 13: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT1XXXxXXX_1_H; break;
7982 case 14: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11400x1050_1; break;
7983 case 15: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11400x1050_1_H; break;
7984 case 16: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11400x1050_2; break;
7985 case 17: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11400x1050_2_H; break;
7986 case 18: LVDSCRT1Ptr = SiS_Pr->SiS_CHTVCRT1UNTSC; break;
7987 case 19: LVDSCRT1Ptr = SiS_Pr->SiS_CHTVCRT1ONTSC; break;
7988 case 20: LVDSCRT1Ptr = SiS_Pr->SiS_CHTVCRT1UPAL; break;
7989 case 21: LVDSCRT1Ptr = SiS_Pr->SiS_CHTVCRT1OPAL; break;
7990 case 22: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT1320x480_1; break; /* FSTN */
7991 case 23: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11024x600_1; break;
7992 case 24: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11024x600_1_H; break;
7993 case 25: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11024x600_2; break;
7994 case 26: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11024x600_2_H; break;
7995 case 27: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11152x768_1; break;
7996 case 28: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11152x768_1_H; break;
7997 case 29: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11152x768_2; break;
7998 case 30: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11152x768_2_H; break;
7999 case 36: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11600x1200_1; break;
8000 case 37: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11600x1200_1_H; break;
8001 case 38: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11600x1200_2; break;
8002 case 39: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11600x1200_2_H; break;
8003 case 40: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11280x768_1; break;
8004 case 41: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11280x768_1_H; break;
8005 case 42: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11280x768_2; break;
8006 case 43: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11280x768_2_H; break;
8007 case 50: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT1640x480_1; break;
8008 case 51: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT1640x480_1_H; break;
8009 case 52: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT1640x480_2; break;
8010 case 53: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT1640x480_2_H; break;
8011 case 54: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT1640x480_3; break;
8012 case 55: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT1640x480_3_H; break;
8013 case 99: LVDSCRT1Ptr = SiS_Pr->SiS_CHTVCRT1SOPAL; break;
8014 default: LVDSCRT1Ptr = SiS_Pr->SiS_LVDSCRT11024x768_1; break;
8015 }
8016
8017 SiS_SetRegAND(SiS_Pr->SiS_P3d4,0x11,0x7f);
8018
8019 tempah = (LVDSCRT1Ptr + ResIndex)->CR[0];
8020 SiS_SetReg(SiS_Pr->SiS_P3d4,0x00,tempah);
8021
8022 for(i=0x02,j=1;i<=0x05;i++,j++){
8023 tempah = (LVDSCRT1Ptr + ResIndex)->CR[j];
8024 SiS_SetReg(SiS_Pr->SiS_P3d4,i,tempah);
8025 }
8026 for(i=0x06,j=5;i<=0x07;i++,j++){
8027 tempah = (LVDSCRT1Ptr + ResIndex)->CR[j];
8028 SiS_SetReg(SiS_Pr->SiS_P3d4,i,tempah);
8029 }
8030 for(i=0x10,j=7;i<=0x11;i++,j++){
8031 tempah = (LVDSCRT1Ptr + ResIndex)->CR[j];
8032 SiS_SetReg(SiS_Pr->SiS_P3d4,i,tempah);
8033 }
8034 for(i=0x15,j=9;i<=0x16;i++,j++){
8035 tempah = (LVDSCRT1Ptr + ResIndex)->CR[j];
8036 SiS_SetReg(SiS_Pr->SiS_P3d4,i,tempah);
8037 }
8038 for(i=0x0A,j=11;i<=0x0C;i++,j++){
8039 tempah = (LVDSCRT1Ptr + ResIndex)->CR[j];
8040 SiS_SetReg(SiS_Pr->SiS_P3c4,i,tempah);
8041 }
8042
8043 tempah = (LVDSCRT1Ptr + ResIndex)->CR[14];
8044 tempah &= 0xE0;
8045 SiS_SetRegANDOR(SiS_Pr->SiS_P3c4,0x0E,0x1f,tempah);
8046
8047 tempah = (LVDSCRT1Ptr + ResIndex)->CR[14];
8048 tempah &= 0x01;
8049 tempah <<= 5;
8050 if(modeflag & DoubleScanMode) tempah |= 0x080;
8051 SiS_SetRegANDOR(SiS_Pr->SiS_P3d4,0x09,~0x020,tempah);
8052}
8053
8054/*********************************************/
8055/* SET CRT2 ECLK */
8056/*********************************************/
8057
8058static void
8059SiS_SetCRT2ECLK(SiS_Private *SiS_Pr, USHORT ModeNo, USHORT ModeIdIndex,
8060 USHORT RefreshRateTableIndex, PSIS_HW_INFO HwInfo)
8061{
8062 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
8063 USHORT clkbase, vclkindex=0;
8064 UCHAR sr2b, sr2c;
8065
8066 if((SiS_Pr->SiS_LCDResInfo == Panel_640x480) || (SiS_Pr->SiS_LCDInfo & LCDPass11)) {
8067 SiS_Pr->SiS_SetFlag &= (~ProgrammingCRT2);
8068 if((SiS_Pr->SiS_RefIndex[RefreshRateTableIndex].Ext_CRTVCLK & 0x3f) == 2) {
8069 RefreshRateTableIndex--;
8070 }
8071 vclkindex = SiS_GetVCLK2Ptr(SiS_Pr, ModeNo, ModeIdIndex,
8072 RefreshRateTableIndex, HwInfo);
8073 SiS_Pr->SiS_SetFlag |= ProgrammingCRT2;
8074 } else {
8075 vclkindex = SiS_GetVCLK2Ptr(SiS_Pr, ModeNo, ModeIdIndex,
8076 RefreshRateTableIndex, HwInfo);
8077 }
8078
8079 sr2b = SiS_Pr->SiS_VCLKData[vclkindex].SR2B;
8080 sr2c = SiS_Pr->SiS_VCLKData[vclkindex].SR2C;
8081
8082 if((SiS_Pr->SiS_CustomT == CUT_BARCO1366) || (SiS_Pr->SiS_CustomT == CUT_BARCO1024)) {
8083 if(SiS_Pr->SiS_UseROM) {
8084 if(ROMAddr[0x220] & 0x01) {
8085 sr2b = ROMAddr[0x227];
8086 sr2c = ROMAddr[0x228];
8087 }
8088 }
8089 }
8090
8091 clkbase = 0x02B;
8092 if(!(SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA)) {
8093 if(!(SiS_Pr->SiS_VBInfo & SetInSlaveMode)) {
8094 clkbase += 3;
8095 }
8096 }
8097
8098 SiS_SetReg(SiS_Pr->SiS_P3c4,0x31,0x20);
8099 SiS_SetReg(SiS_Pr->SiS_P3c4,clkbase,sr2b);
8100 SiS_SetReg(SiS_Pr->SiS_P3c4,clkbase+1,sr2c);
8101 SiS_SetReg(SiS_Pr->SiS_P3c4,0x31,0x10);
8102 SiS_SetReg(SiS_Pr->SiS_P3c4,clkbase,sr2b);
8103 SiS_SetReg(SiS_Pr->SiS_P3c4,clkbase+1,sr2c);
8104 SiS_SetReg(SiS_Pr->SiS_P3c4,0x31,0x00);
8105 SiS_SetReg(SiS_Pr->SiS_P3c4,clkbase,sr2b);
8106 SiS_SetReg(SiS_Pr->SiS_P3c4,clkbase+1,sr2c);
8107}
8108
8109/*********************************************/
8110/* SET UP CHRONTEL CHIPS */
8111/*********************************************/
8112
8113static void
8114SiS_SetCHTVReg(SiS_Private *SiS_Pr, USHORT ModeNo, USHORT ModeIdIndex,
8115 USHORT RefreshRateTableIndex)
8116{
8117#if defined(SIS300) || defined(SIS315H)
8118 USHORT temp, tempbx;
8119#endif
8120 USHORT tempcl;
8121 USHORT TVType, resindex;
8122 const SiS_CHTVRegDataStruct *CHTVRegData = NULL;
8123
8124 if(ModeNo <= 0x13)
8125 tempcl = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_CRT2CRTC;
8126 else
8127 tempcl = SiS_Pr->SiS_RefIndex[RefreshRateTableIndex].Ext_CRT2CRTC;
8128
8129 TVType = 0;
8130 if(SiS_Pr->SiS_TVMode & TVSetCHOverScan) TVType += 1;
8131 if(SiS_Pr->SiS_TVMode & TVSetPAL) {
8132 TVType += 2;
8133 if(SiS_Pr->SiS_ModeType > ModeVGA) {
8134 if(SiS_Pr->SiS_CHSOverScan) TVType = 8;
8135 }
8136 if(SiS_Pr->SiS_TVMode & TVSetPALM) {
8137 TVType = 4;
8138 if(SiS_Pr->SiS_TVMode & TVSetCHOverScan) TVType += 1;
8139 } else if(SiS_Pr->SiS_TVMode & TVSetPALN) {
8140 TVType = 6;
8141 if(SiS_Pr->SiS_TVMode & TVSetCHOverScan) TVType += 1;
8142 }
8143 }
8144 switch(TVType) {
8145 case 0: CHTVRegData = SiS_Pr->SiS_CHTVReg_UNTSC; break;
8146 case 1: CHTVRegData = SiS_Pr->SiS_CHTVReg_ONTSC; break;
8147 case 2: CHTVRegData = SiS_Pr->SiS_CHTVReg_UPAL; break;
8148 case 3: CHTVRegData = SiS_Pr->SiS_CHTVReg_OPAL; break;
8149 case 4: CHTVRegData = SiS_Pr->SiS_CHTVReg_UPALM; break;
8150 case 5: CHTVRegData = SiS_Pr->SiS_CHTVReg_OPALM; break;
8151 case 6: CHTVRegData = SiS_Pr->SiS_CHTVReg_UPALN; break;
8152 case 7: CHTVRegData = SiS_Pr->SiS_CHTVReg_OPALN; break;
8153 case 8: CHTVRegData = SiS_Pr->SiS_CHTVReg_SOPAL; break;
8154 default: CHTVRegData = SiS_Pr->SiS_CHTVReg_OPAL; break;
8155 }
8156 resindex = tempcl & 0x3F;
8157
8158 if(SiS_Pr->SiS_IF_DEF_CH70xx == 1) {
8159
8160#ifdef SIS300
8161
8162 /* Chrontel 7005 - I assume that it does not come with a 315 series chip */
8163
8164 /* We don't support modes >800x600 */
8165 if (resindex > 5) return;
8166
8167 if(SiS_Pr->SiS_TVMode & TVSetPAL) {
8168 SiS_SetCH700x(SiS_Pr,0x4304); /* 0x40=76uA (PAL); 0x03=15bit non-multi RGB*/
8169 SiS_SetCH700x(SiS_Pr,0x6909); /* Black level for PAL (105)*/
8170 } else {
8171 SiS_SetCH700x(SiS_Pr,0x0304); /* upper nibble=71uA (NTSC), 0x03=15bit non-multi RGB*/
8172 SiS_SetCH700x(SiS_Pr,0x7109); /* Black level for NTSC (113)*/
8173 }
8174
8175 temp = CHTVRegData[resindex].Reg[0];
8176 tempbx=((temp&0x00FF)<<8)|0x00; /* Mode register */
8177 SiS_SetCH700x(SiS_Pr,tempbx);
8178 temp = CHTVRegData[resindex].Reg[1];
8179 tempbx=((temp&0x00FF)<<8)|0x07; /* Start active video register */
8180 SiS_SetCH700x(SiS_Pr,tempbx);
8181 temp = CHTVRegData[resindex].Reg[2];
8182 tempbx=((temp&0x00FF)<<8)|0x08; /* Position overflow register */
8183 SiS_SetCH700x(SiS_Pr,tempbx);
8184 temp = CHTVRegData[resindex].Reg[3];
8185 tempbx=((temp&0x00FF)<<8)|0x0A; /* Horiz Position register */
8186 SiS_SetCH700x(SiS_Pr,tempbx);
8187 temp = CHTVRegData[resindex].Reg[4];
8188 tempbx=((temp&0x00FF)<<8)|0x0B; /* Vertical Position register */
8189 SiS_SetCH700x(SiS_Pr,tempbx);
8190
8191 /* Set minimum flicker filter for Luma channel (SR1-0=00),
8192 minimum text enhancement (S3-2=10),
8193 maximum flicker filter for Chroma channel (S5-4=10)
8194 =00101000=0x28 (When reading, S1-0->S3-2, and S3-2->S1-0!)
8195 */
8196 SiS_SetCH700x(SiS_Pr,0x2801);
8197
8198 /* Set video bandwidth
8199 High bandwith Luma composite video filter(S0=1)
8200 low bandwith Luma S-video filter (S2-1=00)
8201 disable peak filter in S-video channel (S3=0)
8202 high bandwidth Chroma Filter (S5-4=11)
8203 =00110001=0x31
8204 */
8205 SiS_SetCH700x(SiS_Pr,0xb103); /* old: 3103 */
8206
8207 /* Register 0x3D does not exist in non-macrovision register map
8208 (Maybe this is a macrovision register?)
8209 */
8210#ifndef SIS_CP
8211 SiS_SetCH70xx(SiS_Pr,0x003D);
8212#endif
8213
8214 /* Register 0x10 only contains 1 writable bit (S0) for sensing,
8215 all other bits a read-only. Macrovision?
8216 */
8217 SiS_SetCH70xxANDOR(SiS_Pr,0x0010,0x1F);
8218
8219 /* Register 0x11 only contains 3 writable bits (S0-S2) for
8220 contrast enhancement (set to 010 -> gain 1 Yout = 17/16*(Yin-30) )
8221 */
8222 SiS_SetCH70xxANDOR(SiS_Pr,0x0211,0xF8);
8223
8224 /* Clear DSEN
8225 */
8226 SiS_SetCH70xxANDOR(SiS_Pr,0x001C,0xEF);
8227
8228 if(!(SiS_Pr->SiS_TVMode & TVSetPAL)) { /* ---- NTSC ---- */
8229 if(SiS_Pr->SiS_TVMode & TVSetCHOverScan) {
8230 if(resindex == 0x04) { /* 640x480 overscan: Mode 16 */
8231 SiS_SetCH70xxANDOR(SiS_Pr,0x0020,0xEF); /* loop filter off */
8232 SiS_SetCH70xxANDOR(SiS_Pr,0x0121,0xFE); /* ACIV on, no need to set FSCI */
8233 } else if(resindex == 0x05) { /* 800x600 overscan: Mode 23 */
8234 SiS_SetCH70xxANDOR(SiS_Pr,0x0118,0xF0); /* 0x18-0x1f: FSCI 469,762,048 */
8235 SiS_SetCH70xxANDOR(SiS_Pr,0x0C19,0xF0);
8236 SiS_SetCH70xxANDOR(SiS_Pr,0x001A,0xF0);
8237 SiS_SetCH70xxANDOR(SiS_Pr,0x001B,0xF0);
8238 SiS_SetCH70xxANDOR(SiS_Pr,0x001C,0xF0);
8239 SiS_SetCH70xxANDOR(SiS_Pr,0x001D,0xF0);
8240 SiS_SetCH70xxANDOR(SiS_Pr,0x001E,0xF0);
8241 SiS_SetCH70xxANDOR(SiS_Pr,0x001F,0xF0);
8242 SiS_SetCH70xxANDOR(SiS_Pr,0x0120,0xEF); /* Loop filter on for mode 23 */
8243 SiS_SetCH70xxANDOR(SiS_Pr,0x0021,0xFE); /* ACIV off, need to set FSCI */
8244 }
8245 } else {
8246 if(resindex == 0x04) { /* ----- 640x480 underscan; Mode 17 */
8247 SiS_SetCH70xxANDOR(SiS_Pr,0x0020,0xEF); /* loop filter off */
8248 SiS_SetCH70xxANDOR(SiS_Pr,0x0121,0xFE);
8249 } else if(resindex == 0x05) { /* ----- 800x600 underscan: Mode 24 */
8250#if 0
8251 SiS_SetCH70xxANDOR(SiS_Pr,0x0118,0xF0); /* (FSCI was 0x1f1c71c7 - this is for mode 22) */
8252 SiS_SetCH70xxANDOR(SiS_Pr,0x0919,0xF0); /* FSCI for mode 24 is 428,554,851 */
8253 SiS_SetCH70xxANDOR(SiS_Pr,0x081A,0xF0); /* 198b3a63 */
8254 SiS_SetCH70xxANDOR(SiS_Pr,0x0b1B,0xF0);
8255 SiS_SetCH70xxANDOR(SiS_Pr,0x041C,0xF0);
8256 SiS_SetCH70xxANDOR(SiS_Pr,0x011D,0xF0);
8257 SiS_SetCH70xxANDOR(SiS_Pr,0x061E,0xF0);
8258 SiS_SetCH70xxANDOR(SiS_Pr,0x051F,0xF0);
8259 SiS_SetCH70xxANDOR(SiS_Pr,0x0020,0xEF); /* loop filter off for mode 24 */
8260 SiS_SetCH70xxANDOR(SiS_Pr,0x0021,0xFE); /* ACIV off, need to set FSCI */
8261#endif /* All alternatives wrong (datasheet wrong?), don't use FSCI */
8262 SiS_SetCH70xxANDOR(SiS_Pr,0x0020,0xEF); /* loop filter off */
8263 SiS_SetCH70xxANDOR(SiS_Pr,0x0121,0xFE);
8264 }
8265 }
8266 } else { /* ---- PAL ---- */
8267 /* We don't play around with FSCI in PAL mode */
8268 if(resindex == 0x04) {
8269 SiS_SetCH70xxANDOR(SiS_Pr,0x0020,0xEF); /* loop filter off */
8270 SiS_SetCH70xxANDOR(SiS_Pr,0x0121,0xFE); /* ACIV on */
8271 } else {
8272 SiS_SetCH70xxANDOR(SiS_Pr,0x0020,0xEF); /* loop filter off */
8273 SiS_SetCH70xxANDOR(SiS_Pr,0x0121,0xFE); /* ACIV on */
8274 }
8275 }
8276
8277#endif /* 300 */
8278
8279 } else {
8280
8281 /* Chrontel 7019 - assumed that it does not come with a 300 series chip */
8282
8283#ifdef SIS315H
8284
8285 /* We don't support modes >1024x768 */
8286 if (resindex > 6) return;
8287
8288 temp = CHTVRegData[resindex].Reg[0];
8289 if(SiS_Pr->SiS_TVMode & TVSetNTSCJ) {
8290 temp |= 0x10;
8291 }
8292 tempbx=((temp & 0x00FF) << 8) | 0x00;
8293 SiS_SetCH701x(SiS_Pr,tempbx);
8294
8295 temp = CHTVRegData[resindex].Reg[1];
8296 tempbx=((temp & 0x00FF) << 8) | 0x01;
8297 SiS_SetCH701x(SiS_Pr,tempbx);
8298
8299 temp = CHTVRegData[resindex].Reg[2];
8300 tempbx=((temp & 0x00FF) << 8) | 0x02;
8301 SiS_SetCH701x(SiS_Pr,tempbx);
8302
8303 temp = CHTVRegData[resindex].Reg[3];
8304 tempbx=((temp & 0x00FF) << 8) | 0x04;
8305 SiS_SetCH701x(SiS_Pr,tempbx);
8306
8307 temp = CHTVRegData[resindex].Reg[4];
8308 tempbx=((temp & 0x00FF) << 8) | 0x03;
8309 SiS_SetCH701x(SiS_Pr,tempbx);
8310
8311 temp = CHTVRegData[resindex].Reg[5];
8312 tempbx=((temp & 0x00FF) << 8) | 0x05;
8313 SiS_SetCH701x(SiS_Pr,tempbx);
8314
8315 temp = CHTVRegData[resindex].Reg[6];
8316 tempbx=((temp & 0x00FF) << 8) | 0x06;
8317 SiS_SetCH701x(SiS_Pr,tempbx);
8318
8319 temp = CHTVRegData[resindex].Reg[7];
8320 if(SiS_Pr->SiS_TVMode & TVSetNTSCJ) {
8321 temp = 0x66;
8322 }
8323 tempbx=((temp & 0x00FF) << 8) | 0x07;
8324 SiS_SetCH701x(SiS_Pr,tempbx);
8325
8326 temp = CHTVRegData[resindex].Reg[8];
8327 tempbx=((temp & 0x00FF) << 8) | 0x08;
8328 SiS_SetCH701x(SiS_Pr,tempbx);
8329
8330 temp = CHTVRegData[resindex].Reg[9];
8331 tempbx=((temp & 0x00FF) << 8) | 0x15;
8332 SiS_SetCH701x(SiS_Pr,tempbx);
8333
8334 temp = CHTVRegData[resindex].Reg[10];
8335 tempbx=((temp & 0x00FF) << 8) | 0x1f;
8336 SiS_SetCH701x(SiS_Pr,tempbx);
8337
8338 temp = CHTVRegData[resindex].Reg[11];
8339 tempbx=((temp & 0x00FF) << 8) | 0x0c;
8340 SiS_SetCH701x(SiS_Pr,tempbx);
8341
8342 temp = CHTVRegData[resindex].Reg[12];
8343 tempbx=((temp & 0x00FF) << 8) | 0x0d;
8344 SiS_SetCH701x(SiS_Pr,tempbx);
8345
8346 temp = CHTVRegData[resindex].Reg[13];
8347 tempbx=((temp & 0x00FF) << 8) | 0x0e;
8348 SiS_SetCH701x(SiS_Pr,tempbx);
8349
8350 temp = CHTVRegData[resindex].Reg[14];
8351 tempbx=((temp & 0x00FF) << 8) | 0x0f;
8352 SiS_SetCH701x(SiS_Pr,tempbx);
8353
8354 temp = CHTVRegData[resindex].Reg[15];
8355 tempbx=((temp & 0x00FF) << 8) | 0x10;
8356 SiS_SetCH701x(SiS_Pr,tempbx);
8357
8358 temp = SiS_GetCH701x(SiS_Pr,0x21) & ~0x02;
8359 /* D1 should be set for PAL, PAL-N and NTSC-J,
8360 but I won't do that for PAL unless somebody
8361 tells me to do so. Since the BIOS uses
8362 non-default CIV values and blacklevels,
8363 this might be compensated anyway.
8364 */
8365 if(SiS_Pr->SiS_TVMode & (TVSetPALN | TVSetNTSCJ)) temp |= 0x02;
8366 SiS_SetCH701x(SiS_Pr,((temp << 8) | 0x21));
8367
8368#endif /* 315 */
8369
8370 }
8371
8372#ifdef SIS_CP
8373 SIS_CP_INIT301_CP3
8374#endif
8375
8376}
8377
8378void
8379SiS_Chrontel701xBLOn(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
8380{
8381 USHORT temp;
8382
8383 /* Enable Chrontel 7019 LCD panel backlight */
8384 if(SiS_Pr->SiS_IF_DEF_CH70xx == 2) {
8385 if(HwInfo->jChipType == SIS_740) {
8386 SiS_SetCH701x(SiS_Pr,0x6566);
8387 } else {
8388 temp = SiS_GetCH701x(SiS_Pr,0x66);
8389 temp |= 0x20;
8390 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x66);
8391 }
8392 }
8393}
8394
8395void
8396SiS_Chrontel701xBLOff(SiS_Private *SiS_Pr)
8397{
8398 USHORT temp;
8399
8400 /* Disable Chrontel 7019 LCD panel backlight */
8401 if(SiS_Pr->SiS_IF_DEF_CH70xx == 2) {
8402 temp = SiS_GetCH701x(SiS_Pr,0x66);
8403 temp &= 0xDF;
8404 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x66);
8405 }
8406}
8407
8408#ifdef SIS315H /* ----------- 315 series only ---------- */
8409
8410static void
8411SiS_ChrontelPowerSequencing(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
8412{
8413 UCHAR regtable[] = { 0x67, 0x68, 0x69, 0x6a, 0x6b };
8414 UCHAR table1024_740[] = { 0x01, 0x02, 0x01, 0x01, 0x01 };
8415 UCHAR table1400_740[] = { 0x01, 0x6e, 0x01, 0x01, 0x01 };
8416 UCHAR asus1024_740[] = { 0x19, 0x6e, 0x01, 0x19, 0x09 };
8417 UCHAR asus1400_740[] = { 0x19, 0x6e, 0x01, 0x19, 0x09 };
8418 UCHAR table1024_650[] = { 0x01, 0x02, 0x01, 0x01, 0x02 };
8419 UCHAR table1400_650[] = { 0x01, 0x02, 0x01, 0x01, 0x02 };
8420 UCHAR *tableptr = NULL;
8421 int i;
8422
8423 /* Set up Power up/down timing */
8424
8425 if(HwInfo->jChipType == SIS_740) {
8426 if(SiS_Pr->SiS_LCDResInfo == Panel_1024x768) {
8427 if(SiS_Pr->SiS_CustomT == CUT_ASUSL3000D) tableptr = asus1024_740;
8428 else tableptr = table1024_740;
8429 } else if((SiS_Pr->SiS_LCDResInfo == Panel_1280x1024) ||
8430 (SiS_Pr->SiS_LCDResInfo == Panel_1400x1050) ||
8431 (SiS_Pr->SiS_LCDResInfo == Panel_1600x1200)) {
8432 if(SiS_Pr->SiS_CustomT == CUT_ASUSL3000D) tableptr = asus1400_740;
8433 else tableptr = table1400_740;
8434 } else return;
8435 } else {
8436 if(SiS_Pr->SiS_LCDResInfo == Panel_1024x768) {
8437 tableptr = table1024_650;
8438 } else if((SiS_Pr->SiS_LCDResInfo == Panel_1280x1024) ||
8439 (SiS_Pr->SiS_LCDResInfo == Panel_1400x1050) ||
8440 (SiS_Pr->SiS_LCDResInfo == Panel_1600x1200)) {
8441 tableptr = table1400_650;
8442 } else return;
8443 }
8444
8445 for(i=0; i<5; i++) {
8446 SiS_SetCH701x(SiS_Pr,(tableptr[i] << 8) | regtable[i]);
8447 }
8448}
8449
8450static void
8451SiS_SetCH701xForLCD(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
8452{
8453 UCHAR regtable[] = { 0x1c, 0x5f, 0x64, 0x6f, 0x70, 0x71,
8454 0x72, 0x73, 0x74, 0x76, 0x78, 0x7d, 0x66 };
8455 UCHAR table1024_740[] = { 0x60, 0x02, 0x00, 0x07, 0x40, 0xed,
8456 0xa3, 0xc8, 0xc7, 0xac, 0xe0, 0x02, 0x44 };
8457 UCHAR table1280_740[] = { 0x60, 0x03, 0x11, 0x00, 0x40, 0xe3,
8458 0xad, 0xdb, 0xf6, 0xac, 0xe0, 0x02, 0x44 };
8459 UCHAR table1400_740[] = { 0x60, 0x03, 0x11, 0x00, 0x40, 0xe3,
8460 0xad, 0xdb, 0xf6, 0xac, 0xe0, 0x02, 0x44 };
8461 UCHAR table1600_740[] = { 0x60, 0x04, 0x11, 0x00, 0x40, 0xe3,
8462 0xad, 0xde, 0xf6, 0xac, 0x60, 0x1a, 0x44 };
8463 UCHAR table1024_650[] = { 0x60, 0x02, 0x00, 0x07, 0x40, 0xed,
8464 0xa3, 0xc8, 0xc7, 0xac, 0x60, 0x02 };
8465 UCHAR table1280_650[] = { 0x60, 0x03, 0x11, 0x00, 0x40, 0xe3,
8466 0xad, 0xdb, 0xf6, 0xac, 0xe0, 0x02 };
8467 UCHAR table1400_650[] = { 0x60, 0x03, 0x11, 0x00, 0x40, 0xef,
8468 0xad, 0xdb, 0xf6, 0xac, 0x60, 0x02 };
8469 UCHAR table1600_650[] = { 0x60, 0x04, 0x11, 0x00, 0x40, 0xe3,
8470 0xad, 0xde, 0xf6, 0xac, 0x60, 0x1a };
8471 UCHAR *tableptr = NULL;
8472 USHORT tempbh;
8473 int i;
8474
8475 if(HwInfo->jChipType == SIS_740) {
8476 if(SiS_Pr->SiS_LCDResInfo == Panel_1024x768) tableptr = table1024_740;
8477 else if(SiS_Pr->SiS_LCDResInfo == Panel_1280x1024) tableptr = table1280_740;
8478 else if(SiS_Pr->SiS_LCDResInfo == Panel_1400x1050) tableptr = table1400_740;
8479 else if(SiS_Pr->SiS_LCDResInfo == Panel_1600x1200) tableptr = table1600_740;
8480 else return;
8481 } else {
8482 if(SiS_Pr->SiS_LCDResInfo == Panel_1024x768) tableptr = table1024_650;
8483 else if(SiS_Pr->SiS_LCDResInfo == Panel_1280x1024) tableptr = table1280_650;
8484 else if(SiS_Pr->SiS_LCDResInfo == Panel_1400x1050) tableptr = table1400_650;
8485 else if(SiS_Pr->SiS_LCDResInfo == Panel_1600x1200) tableptr = table1600_650;
8486 else return;
8487 }
8488
8489 tempbh = SiS_GetCH701x(SiS_Pr,0x74);
8490 if((tempbh == 0xf6) || (tempbh == 0xc7)) {
8491 tempbh = SiS_GetCH701x(SiS_Pr,0x73);
8492 if(tempbh == 0xc8) {
8493 if(SiS_Pr->SiS_LCDResInfo == Panel_1024x768) return;
8494 } else if(tempbh == 0xdb) {
8495 if(SiS_Pr->SiS_LCDResInfo == Panel_1280x1024) return;
8496 if(SiS_Pr->SiS_LCDResInfo == Panel_1400x1050) return;
8497 } else if(tempbh == 0xde) {
8498 if(SiS_Pr->SiS_LCDResInfo == Panel_1600x1200) return;
8499 }
8500 }
8501
8502 if(HwInfo->jChipType == SIS_740) tempbh = 0x0d;
8503 else tempbh = 0x0c;
8504
8505 for(i = 0; i < tempbh; i++) {
8506 SiS_SetCH701x(SiS_Pr,(tableptr[i] << 8) | regtable[i]);
8507 }
8508 SiS_ChrontelPowerSequencing(SiS_Pr,HwInfo);
8509 tempbh = SiS_GetCH701x(SiS_Pr,0x1e);
8510 tempbh |= 0xc0;
8511 SiS_SetCH701x(SiS_Pr,(tempbh << 8) | 0x1e);
8512
8513 if(HwInfo->jChipType == SIS_740) {
8514 tempbh = SiS_GetCH701x(SiS_Pr,0x1c);
8515 tempbh &= 0xfb;
8516 SiS_SetCH701x(SiS_Pr,(tempbh << 8) | 0x1c);
8517 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x2d,0x03);
8518 tempbh = SiS_GetCH701x(SiS_Pr,0x64);
8519 tempbh |= 0x40;
8520 SiS_SetCH701x(SiS_Pr,(tempbh << 8) | 0x64);
8521 tempbh = SiS_GetCH701x(SiS_Pr,0x03);
8522 tempbh &= 0x3f;
8523 SiS_SetCH701x(SiS_Pr,(tempbh << 8) | 0x03);
8524 }
8525}
8526
8527static void
8528SiS_ChrontelResetVSync(SiS_Private *SiS_Pr)
8529{
8530 unsigned char temp, temp1;
8531
8532 temp1 = SiS_GetCH701x(SiS_Pr,0x49);
8533 SiS_SetCH701x(SiS_Pr,0x3e49);
8534 temp = SiS_GetCH701x(SiS_Pr,0x47);
8535 temp &= 0x7f; /* Use external VSYNC */
8536 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x47);
8537 SiS_LongDelay(SiS_Pr,3);
8538 temp = SiS_GetCH701x(SiS_Pr,0x47);
8539 temp |= 0x80; /* Use internal VSYNC */
8540 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x47);
8541 SiS_SetCH701x(SiS_Pr,(temp1 << 8) | 0x49);
8542}
8543
8544static void
8545SiS_Chrontel701xOn(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
8546{
8547 USHORT temp;
8548
8549 if(SiS_Pr->SiS_IF_DEF_CH70xx == 2) {
8550 if(HwInfo->jChipType == SIS_740) {
8551 temp = SiS_GetCH701x(SiS_Pr,0x1c);
8552 temp |= 0x04; /* Invert XCLK phase */
8553 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x1c);
8554 }
8555 if(SiS_IsYPbPr(SiS_Pr, HwInfo)) {
8556 temp = SiS_GetCH701x(SiS_Pr,0x01);
8557 temp &= 0x3f;
8558 temp |= 0x80; /* Enable YPrPb (HDTV) */
8559 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x01);
8560 }
8561 if(SiS_IsChScart(SiS_Pr, HwInfo)) {
8562 temp = SiS_GetCH701x(SiS_Pr,0x01);
8563 temp &= 0x3f;
8564 temp |= 0xc0; /* Enable SCART + CVBS */
8565 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x01);
8566 }
8567 if(HwInfo->jChipType == SIS_740) {
8568 SiS_ChrontelResetVSync(SiS_Pr);
8569 SiS_SetCH701x(SiS_Pr,0x2049); /* Enable TV path */
8570 } else {
8571 SiS_SetCH701x(SiS_Pr,0x2049); /* Enable TV path */
8572 temp = SiS_GetCH701x(SiS_Pr,0x49);
8573 if(SiS_IsYPbPr(SiS_Pr,HwInfo)) {
8574 temp = SiS_GetCH701x(SiS_Pr,0x73);
8575 temp |= 0x60;
8576 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x73);
8577 }
8578 temp = SiS_GetCH701x(SiS_Pr,0x47);
8579 temp &= 0x7f;
8580 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x47);
8581 SiS_LongDelay(SiS_Pr,2);
8582 temp = SiS_GetCH701x(SiS_Pr,0x47);
8583 temp |= 0x80;
8584 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x47);
8585 }
8586 }
8587}
8588
8589static void
8590SiS_Chrontel701xOff(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
8591{
8592 USHORT temp;
8593
8594 /* Complete power down of LVDS */
8595 if(SiS_Pr->SiS_IF_DEF_CH70xx == 2) {
8596 if(HwInfo->jChipType == SIS_740) {
8597 SiS_LongDelay(SiS_Pr,1);
8598 SiS_GenericDelay(SiS_Pr,0x16ff);
8599 SiS_SetCH701x(SiS_Pr,0xac76);
8600 SiS_SetCH701x(SiS_Pr,0x0066);
8601 } else {
8602 SiS_LongDelay(SiS_Pr,2);
8603 temp = SiS_GetCH701x(SiS_Pr,0x76);
8604 temp &= 0xfc;
8605 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x76);
8606 SiS_SetCH701x(SiS_Pr,0x0066);
8607 }
8608 }
8609}
8610
8611static void
8612SiS_ChrontelResetDB(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
8613{
8614 USHORT temp;
8615
8616 if(HwInfo->jChipType == SIS_740) {
8617
8618 temp = SiS_GetCH701x(SiS_Pr,0x4a); /* Version ID */
8619 temp &= 0x01;
8620 if(!temp) {
8621
8622 if(SiS_WeHaveBacklightCtrl(SiS_Pr,HwInfo)) {
8623 temp = SiS_GetCH701x(SiS_Pr,0x49);
8624 SiS_SetCH701x(SiS_Pr,0x3e49);
8625 }
8626 /* Reset Chrontel 7019 datapath */
8627 SiS_SetCH701x(SiS_Pr,0x1048);
8628 SiS_LongDelay(SiS_Pr,1);
8629 SiS_SetCH701x(SiS_Pr,0x1848);
8630
8631 if(SiS_WeHaveBacklightCtrl(SiS_Pr, HwInfo)) {
8632 SiS_ChrontelResetVSync(SiS_Pr);
8633 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x49);
8634 }
8635
8636 } else {
8637
8638 /* Clear/set/clear GPIO */
8639 temp = SiS_GetCH701x(SiS_Pr,0x5c);
8640 temp &= 0xef;
8641 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x5c);
8642 temp = SiS_GetCH701x(SiS_Pr,0x5c);
8643 temp |= 0x10;
8644 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x5c);
8645 temp = SiS_GetCH701x(SiS_Pr,0x5c);
8646 temp &= 0xef;
8647 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x5c);
8648 temp = SiS_GetCH701x(SiS_Pr,0x61);
8649 if(!temp) {
8650 SiS_SetCH701xForLCD(SiS_Pr, HwInfo);
8651 }
8652 }
8653
8654 } else { /* 650 */
8655 /* Reset Chrontel 7019 datapath */
8656 SiS_SetCH701x(SiS_Pr,0x1048);
8657 SiS_LongDelay(SiS_Pr,1);
8658 SiS_SetCH701x(SiS_Pr,0x1848);
8659 }
8660}
8661
8662static void
8663SiS_ChrontelInitTVVSync(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
8664{
8665 USHORT temp;
8666
8667 if(HwInfo->jChipType == SIS_740) {
8668
8669 if(SiS_WeHaveBacklightCtrl(SiS_Pr,HwInfo)) {
8670 SiS_ChrontelResetVSync(SiS_Pr);
8671 }
8672
8673 } else {
8674
8675 SiS_SetCH701x(SiS_Pr,0xaf76); /* Power up LVDS block */
8676 temp = SiS_GetCH701x(SiS_Pr,0x49);
8677 temp &= 1;
8678 if(temp != 1) { /* TV block powered? (0 = yes, 1 = no) */
8679 temp = SiS_GetCH701x(SiS_Pr,0x47);
8680 temp &= 0x70;
8681 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x47); /* enable VSYNC */
8682 SiS_LongDelay(SiS_Pr,3);
8683 temp = SiS_GetCH701x(SiS_Pr,0x47);
8684 temp |= 0x80;
8685 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x47); /* disable VSYNC */
8686 }
8687
8688 }
8689}
8690
8691static void
8692SiS_ChrontelDoSomething3(SiS_Private *SiS_Pr, USHORT ModeNo, PSIS_HW_INFO HwInfo)
8693{
8694 USHORT temp,temp1;
8695
8696 if(HwInfo->jChipType == SIS_740) {
8697
8698 temp = SiS_GetCH701x(SiS_Pr,0x61);
8699 if(temp < 1) {
8700 temp++;
8701 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x61);
8702 }
8703 SiS_SetCH701x(SiS_Pr,0x4566); /* Panel power on */
8704 SiS_SetCH701x(SiS_Pr,0xaf76); /* All power on */
8705 SiS_LongDelay(SiS_Pr,1);
8706 SiS_GenericDelay(SiS_Pr,0x16ff);
8707
8708 } else { /* 650 */
8709
8710 temp1 = 0;
8711 temp = SiS_GetCH701x(SiS_Pr,0x61);
8712 if(temp < 2) {
8713 temp++;
8714 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x61);
8715 temp1 = 1;
8716 }
8717 SiS_SetCH701x(SiS_Pr,0xac76);
8718 temp = SiS_GetCH701x(SiS_Pr,0x66);
8719 temp |= 0x5f;
8720 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x66);
8721 if(ModeNo > 0x13) {
8722 if(SiS_WeHaveBacklightCtrl(SiS_Pr, HwInfo)) {
8723 SiS_GenericDelay(SiS_Pr,0x3ff);
8724 } else {
8725 SiS_GenericDelay(SiS_Pr,0x2ff);
8726 }
8727 } else {
8728 if(!temp1)
8729 SiS_GenericDelay(SiS_Pr,0x2ff);
8730 }
8731 temp = SiS_GetCH701x(SiS_Pr,0x76);
8732 temp |= 0x03;
8733 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x76);
8734 temp = SiS_GetCH701x(SiS_Pr,0x66);
8735 temp &= 0x7f;
8736 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x66);
8737 SiS_LongDelay(SiS_Pr,1);
8738
8739 }
8740}
8741
8742static void
8743SiS_ChrontelDoSomething2(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
8744{
8745 USHORT temp,tempcl,tempch;
8746
8747 SiS_LongDelay(SiS_Pr, 1);
8748 tempcl = 3;
8749 tempch = 0;
8750
8751 do {
8752 temp = SiS_GetCH701x(SiS_Pr,0x66);
8753 temp &= 0x04; /* PLL stable? -> bail out */
8754 if(temp == 0x04) break;
8755
8756 if(HwInfo->jChipType == SIS_740) {
8757 /* Power down LVDS output, PLL normal operation */
8758 SiS_SetCH701x(SiS_Pr,0xac76);
8759 }
8760
8761 SiS_SetCH701xForLCD(SiS_Pr,HwInfo);
8762
8763 if(tempcl == 0) {
8764 if(tempch == 3) break;
8765 SiS_ChrontelResetDB(SiS_Pr,HwInfo);
8766 tempcl = 3;
8767 tempch++;
8768 }
8769 tempcl--;
8770 temp = SiS_GetCH701x(SiS_Pr,0x76);
8771 temp &= 0xfb; /* Reset PLL */
8772 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x76);
8773 SiS_LongDelay(SiS_Pr,2);
8774 temp = SiS_GetCH701x(SiS_Pr,0x76);
8775 temp |= 0x04; /* PLL normal operation */
8776 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x76);
8777 if(HwInfo->jChipType == SIS_740) {
8778 SiS_SetCH701x(SiS_Pr,0xe078); /* PLL loop filter */
8779 } else {
8780 SiS_SetCH701x(SiS_Pr,0x6078);
8781 }
8782 SiS_LongDelay(SiS_Pr,2);
8783 } while(0);
8784
8785 SiS_SetCH701x(SiS_Pr,0x0077); /* MV? */
8786}
8787
8788static void
8789SiS_ChrontelDoSomething1(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
8790{
8791 USHORT temp;
8792
8793 temp = SiS_GetCH701x(SiS_Pr,0x03);
8794 temp |= 0x80; /* Set datapath 1 to TV */
8795 temp &= 0xbf; /* Set datapath 2 to LVDS */
8796 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x03);
8797
8798 if(HwInfo->jChipType == SIS_740) {
8799
8800 temp = SiS_GetCH701x(SiS_Pr,0x1c);
8801 temp &= 0xfb; /* Normal XCLK phase */
8802 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x1c);
8803
8804 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x2d,0x03);
8805
8806 temp = SiS_GetCH701x(SiS_Pr,0x64);
8807 temp |= 0x40; /* ? Bit not defined */
8808 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x64);
8809
8810 temp = SiS_GetCH701x(SiS_Pr,0x03);
8811 temp &= 0x3f; /* D1 input to both LVDS and TV */
8812 SiS_SetCH701x(SiS_Pr,(temp << 8) | 0x03);
8813
8814 if(SiS_Pr->SiS_CustomT == CUT_ASUSL3000D) {
8815 SiS_SetCH701x(SiS_Pr,0x4063); /* LVDS off */
8816 SiS_LongDelay(SiS_Pr, 1);
8817 SiS_SetCH701x(SiS_Pr,0x0063); /* LVDS on */
8818 SiS_ChrontelResetDB(SiS_Pr, HwInfo);
8819 SiS_ChrontelDoSomething2(SiS_Pr, HwInfo);
8820 SiS_ChrontelDoSomething3(SiS_Pr, 0, HwInfo);
8821 } else {
8822 temp = SiS_GetCH701x(SiS_Pr,0x66);
8823 if(temp != 0x45) {
8824 SiS_ChrontelResetDB(SiS_Pr, HwInfo);
8825 SiS_ChrontelDoSomething2(SiS_Pr, HwInfo);
8826 SiS_ChrontelDoSomething3(SiS_Pr, 0, HwInfo);
8827 }
8828 }
8829
8830 } else { /* 650 */
8831
8832 SiS_ChrontelResetDB(SiS_Pr,HwInfo);
8833 SiS_ChrontelDoSomething2(SiS_Pr,HwInfo);
8834 temp = SiS_GetReg(SiS_Pr->SiS_P3d4,0x34);
8835 SiS_ChrontelDoSomething3(SiS_Pr,temp,HwInfo);
8836 SiS_SetCH701x(SiS_Pr,0xaf76); /* All power on, LVDS normal operation */
8837
8838 }
8839
8840}
8841#endif /* 315 series */
8842
8843/*********************************************/
8844/* MAIN: SET CRT2 REGISTER GROUP */
8845/*********************************************/
8846
8847BOOLEAN
8848SiS_SetCRT2Group(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo, USHORT ModeNo)
8849{
8850#ifdef SIS300
8851 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
8852#endif
8853 USHORT ModeIdIndex, RefreshRateTableIndex;
8854#if 0
8855 USHORT temp;
8856#endif
8857
8858 SiS_Pr->SiS_SetFlag |= ProgrammingCRT2;
8859
8860 if(!SiS_Pr->UseCustomMode) {
8861 SiS_SearchModeID(SiS_Pr, &ModeNo, &ModeIdIndex);
8862 } else {
8863 ModeIdIndex = 0;
8864 }
8865
8866 /* Used for shifting CR33 */
8867 SiS_Pr->SiS_SelectCRT2Rate = 4;
8868
8869 SiS_UnLockCRT2(SiS_Pr, HwInfo);
8870
8871 RefreshRateTableIndex = SiS_GetRatePtr(SiS_Pr, ModeNo, ModeIdIndex, HwInfo);
8872
8873 SiS_SaveCRT2Info(SiS_Pr,ModeNo);
8874
8875 if(SiS_Pr->SiS_SetFlag & LowModeTests) {
8876 SiS_DisableBridge(SiS_Pr,HwInfo);
8877 if((SiS_Pr->SiS_IF_DEF_LVDS == 1) && (HwInfo->jChipType == SIS_730)) {
8878 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x00,0x80);
8879 }
8880 SiS_SetCRT2ModeRegs(SiS_Pr, ModeNo, ModeIdIndex, HwInfo);
8881 }
8882
8883 if(SiS_Pr->SiS_VBInfo & DisableCRT2Display) {
8884 SiS_LockCRT2(SiS_Pr, HwInfo);
8885 SiS_DisplayOn(SiS_Pr);
8886 return TRUE;
8887 }
8888
8889 SiS_GetCRT2Data(SiS_Pr, ModeNo, ModeIdIndex, RefreshRateTableIndex, HwInfo);
8890
8891 /* Set up Panel Link for LVDS and LCDA */
8892 SiS_Pr->SiS_LCDHDES = SiS_Pr->SiS_LCDVDES = 0;
8893 if( (SiS_Pr->SiS_IF_DEF_LVDS == 1) ||
8894 ((SiS_Pr->SiS_VBType & VB_NoLCD) && (SiS_Pr->SiS_VBInfo & SetCRT2ToLCD)) ||
8895 ((HwInfo->jChipType >= SIS_315H) && (SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV)) ) {
8896 SiS_GetLVDSDesData(SiS_Pr, ModeNo, ModeIdIndex, RefreshRateTableIndex, HwInfo);
8897 }
8898
8899#ifdef LINUX_XF86
8900#ifdef TWDEBUG
8901 xf86DrvMsg(0, X_INFO, "(init301: LCDHDES 0x%03x LCDVDES 0x%03x)\n", SiS_Pr->SiS_LCDHDES, SiS_Pr->SiS_LCDVDES);
8902 xf86DrvMsg(0, X_INFO, "(init301: HDE 0x%03x VDE 0x%03x)\n", SiS_Pr->SiS_HDE, SiS_Pr->SiS_VDE);
8903 xf86DrvMsg(0, X_INFO, "(init301: VGAHDE 0x%03x VGAVDE 0x%03x)\n", SiS_Pr->SiS_VGAHDE, SiS_Pr->SiS_VGAVDE);
8904 xf86DrvMsg(0, X_INFO, "(init301: HT 0x%03x VT 0x%03x)\n", SiS_Pr->SiS_HT, SiS_Pr->SiS_VT);
8905 xf86DrvMsg(0, X_INFO, "(init301: VGAHT 0x%03x VGAVT 0x%03x)\n", SiS_Pr->SiS_VGAHT, SiS_Pr->SiS_VGAVT);
8906#endif
8907#endif
8908
8909 if(SiS_Pr->SiS_SetFlag & LowModeTests) {
8910 SiS_SetGroup1(SiS_Pr, ModeNo, ModeIdIndex, HwInfo, RefreshRateTableIndex);
8911 }
8912
8913 if(SiS_Pr->SiS_VBType & VB_SISVB) {
8914
8915 if(SiS_Pr->SiS_SetFlag & LowModeTests) {
8916
8917 SiS_SetGroup2(SiS_Pr, ModeNo, ModeIdIndex, RefreshRateTableIndex, HwInfo);
8918#ifdef SIS315H
8919 SiS_SetGroup2_C_ELV(SiS_Pr, ModeNo, ModeIdIndex, RefreshRateTableIndex, HwInfo);
8920#endif
8921 SiS_SetGroup3(SiS_Pr, ModeNo, ModeIdIndex, HwInfo);
8922 SiS_SetGroup4(SiS_Pr, ModeNo, ModeIdIndex, RefreshRateTableIndex, HwInfo);
8923#ifdef SIS315H
8924 SiS_SetGroup4_C_ELV(SiS_Pr, HwInfo, ModeNo, ModeIdIndex);
8925#endif
8926 SiS_SetGroup5(SiS_Pr, ModeNo, ModeIdIndex, HwInfo);
8927
8928 SiS_SetCRT2Sync(SiS_Pr, ModeNo, RefreshRateTableIndex, HwInfo);
8929
8930 /* For 301BDH (Panel link initialization): */
8931 if((SiS_Pr->SiS_VBType & VB_NoLCD) && (SiS_Pr->SiS_VBInfo & SetCRT2ToLCD)) {
8932 if(SiS_Pr->SiS_LCDResInfo != Panel_640x480) {
8933 if(!((SiS_Pr->SiS_SetFlag & SetDOSMode) && ((ModeNo == 0x03) || (ModeNo == 0x10)))) {
8934 if(SiS_Pr->SiS_VBInfo & SetInSlaveMode) {
8935 SiS_ModCRT1CRTC(SiS_Pr,ModeNo,ModeIdIndex,
8936 RefreshRateTableIndex,HwInfo);
8937 }
8938 }
8939 }
8940 SiS_SetCRT2ECLK(SiS_Pr,ModeNo,ModeIdIndex,
8941 RefreshRateTableIndex,HwInfo);
8942 }
8943 }
8944
8945 } else {
8946
8947 SiS_SetCRT2Sync(SiS_Pr, ModeNo, RefreshRateTableIndex, HwInfo);
8948
8949 if(!(SiS_Pr->SiS_LCDInfo & LCDPass11)) {
8950 SiS_ModCRT1CRTC(SiS_Pr,ModeNo,ModeIdIndex,RefreshRateTableIndex,HwInfo);
8951 }
8952
8953 SiS_SetCRT2ECLK(SiS_Pr,ModeNo,ModeIdIndex,RefreshRateTableIndex,HwInfo);
8954
8955 if(SiS_Pr->SiS_SetFlag & LowModeTests) {
8956 if(SiS_Pr->SiS_IF_DEF_CH70xx != 0) {
8957 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA)) {
8958 if(SiS_Pr->SiS_IF_DEF_CH70xx == 2) {
8959#ifdef SIS315H
8960 SiS_SetCH701xForLCD(SiS_Pr,HwInfo);
8961#endif
8962 }
8963 }
8964 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) {
8965 SiS_SetCHTVReg(SiS_Pr,ModeNo,ModeIdIndex,RefreshRateTableIndex);
8966 }
8967 }
8968 }
8969
8970 }
8971
8972#ifdef SIS300
8973 if(HwInfo->jChipType < SIS_315H) {
8974 if(SiS_Pr->SiS_SetFlag & LowModeTests) {
8975 if(SiS_Pr->SiS_UseOEM) {
8976 if((SiS_Pr->SiS_UseROM) && (SiS_Pr->SiS_UseOEM == -1)) {
8977 if((ROMAddr[0x233] == 0x12) && (ROMAddr[0x234] == 0x34)) {
8978 SiS_OEM300Setting(SiS_Pr,HwInfo,ModeNo,ModeIdIndex,
8979 RefreshRateTableIndex);
8980 }
8981 } else {
8982 SiS_OEM300Setting(SiS_Pr,HwInfo,ModeNo,ModeIdIndex,
8983 RefreshRateTableIndex);
8984 }
8985 }
8986 if(SiS_Pr->SiS_IF_DEF_LVDS == 1) {
8987 if((SiS_Pr->SiS_CustomT == CUT_BARCO1366) ||
8988 (SiS_Pr->SiS_CustomT == CUT_BARCO1024)) {
8989 SetOEMLCDData2(SiS_Pr, HwInfo, ModeNo, ModeIdIndex,RefreshRateTableIndex);
8990 }
8991 SiS_DisplayOn(SiS_Pr);
8992 }
8993 }
8994 }
8995#endif
8996
8997#ifdef SIS315H
8998 if(HwInfo->jChipType >= SIS_315H) {
8999 if(SiS_Pr->SiS_SetFlag & LowModeTests) {
9000 if(HwInfo->jChipType < SIS_661) {
9001 SiS_FinalizeLCD(SiS_Pr, ModeNo, ModeIdIndex, HwInfo);
9002 SiS_OEM310Setting(SiS_Pr, HwInfo, ModeNo, ModeIdIndex, RefreshRateTableIndex);
9003 } else {
9004 SiS_OEM661Setting(SiS_Pr, HwInfo, ModeNo, ModeIdIndex, RefreshRateTableIndex);
9005 }
9006 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x01,0x40);
9007 }
9008 }
9009#endif
9010
9011 if(SiS_Pr->SiS_SetFlag & LowModeTests) {
9012 SiS_EnableBridge(SiS_Pr, HwInfo);
9013 }
9014
9015 SiS_DisplayOn(SiS_Pr);
9016
9017 if(SiS_Pr->SiS_IF_DEF_CH70xx == 1) {
9018 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) {
9019 /* Disable LCD panel when using TV */
9020 SiS_SetRegSR11ANDOR(SiS_Pr,HwInfo,0xFF,0x0C);
9021 } else {
9022 /* Disable TV when using LCD */
9023 SiS_SetCH70xxANDOR(SiS_Pr,0x010E,0xF8);
9024 }
9025 }
9026
9027 if(SiS_Pr->SiS_SetFlag & LowModeTests) {
9028 SiS_LockCRT2(SiS_Pr,HwInfo);
9029 }
9030
9031 return TRUE;
9032}
9033
9034
9035/*********************************************/
9036/* ENABLE/DISABLE LCD BACKLIGHT (SIS) */
9037/*********************************************/
9038
9039void
9040SiS_SiS30xBLOn(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
9041{
9042 /* Switch on LCD backlight on SiS30xLV */
9043 SiS_DDC2Delay(SiS_Pr,0xff00);
9044 if(!(SiS_GetReg(SiS_Pr->SiS_Part4Port,0x26) & 0x02)) {
9045 SiS_SetRegOR(SiS_Pr->SiS_Part4Port,0x26,0x02);
9046 SiS_WaitVBRetrace(SiS_Pr,HwInfo);
9047 }
9048 if(!(SiS_GetReg(SiS_Pr->SiS_Part4Port,0x26) & 0x01)) {
9049 SiS_SetRegOR(SiS_Pr->SiS_Part4Port,0x26,0x01);
9050 }
9051}
9052
9053void
9054SiS_SiS30xBLOff(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
9055{
9056 /* Switch off LCD backlight on SiS30xLV */
9057 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x26,0xFE);
9058 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x26,0xFD);
9059 SiS_DDC2Delay(SiS_Pr,0xe000);
9060}
9061
9062/*********************************************/
9063/* DDC RELATED FUNCTIONS */
9064/*********************************************/
9065
9066static void
9067SiS_SetupDDCN(SiS_Private *SiS_Pr)
9068{
9069 SiS_Pr->SiS_DDC_NData = ~SiS_Pr->SiS_DDC_Data;
9070 SiS_Pr->SiS_DDC_NClk = ~SiS_Pr->SiS_DDC_Clk;
9071 if((SiS_Pr->SiS_DDC_Index == 0x11) && (SiS_Pr->SiS_SensibleSR11)) {
9072 SiS_Pr->SiS_DDC_NData &= 0x0f;
9073 SiS_Pr->SiS_DDC_NClk &= 0x0f;
9074 }
9075}
9076
9077#ifdef SIS300
9078static UCHAR *
9079SiS_SetTrumpBlockLoop(SiS_Private *SiS_Pr, UCHAR *dataptr)
9080{
9081 int i, j, num;
9082 USHORT tempah,temp;
9083 UCHAR *mydataptr;
9084
9085 for(i=0; i<20; i++) { /* Do 20 attempts to write */
9086 mydataptr = dataptr;
9087 num = *mydataptr++;
9088 if(!num) return mydataptr;
9089 if(i) {
9090 SiS_SetStop(SiS_Pr);
9091 SiS_DDC2Delay(SiS_Pr,SiS_I2CDELAYSHORT*2);
9092 }
9093 if(SiS_SetStart(SiS_Pr)) continue; /* Set start condition */
9094 tempah = SiS_Pr->SiS_DDC_DeviceAddr;
9095 temp = SiS_WriteDDC2Data(SiS_Pr,tempah); /* Write DAB (S0=0=write) */
9096 if(temp) continue; /* (ERROR: no ack) */
9097 tempah = *mydataptr++;
9098 temp = SiS_WriteDDC2Data(SiS_Pr,tempah); /* Write register number */
9099 if(temp) continue; /* (ERROR: no ack) */
9100 for(j=0; j<num; j++) {
9101 tempah = *mydataptr++;
9102 temp = SiS_WriteDDC2Data(SiS_Pr,tempah);/* Write DAB (S0=0=write) */
9103 if(temp) break;
9104 }
9105 if(temp) continue;
9106 if(SiS_SetStop(SiS_Pr)) continue;
9107 return mydataptr;
9108 }
9109 return NULL;
9110}
9111
9112static BOOLEAN
9113SiS_SetTrumpionBlock(SiS_Private *SiS_Pr, UCHAR *dataptr)
9114{
9115 SiS_Pr->SiS_DDC_DeviceAddr = 0xF0; /* DAB (Device Address Byte) */
9116 SiS_Pr->SiS_DDC_Index = 0x11; /* Bit 0 = SC; Bit 1 = SD */
9117 SiS_Pr->SiS_DDC_Data = 0x02; /* Bitmask in IndexReg for Data */
9118 SiS_Pr->SiS_DDC_Clk = 0x01; /* Bitmask in IndexReg for Clk */
9119 SiS_SetupDDCN(SiS_Pr);
9120
9121 SiS_SetSwitchDDC2(SiS_Pr);
9122
9123 while(*dataptr) {
9124 dataptr = SiS_SetTrumpBlockLoop(SiS_Pr, dataptr);
9125 if(!dataptr) return FALSE;
9126 }
9127#ifdef TWDEBUG
9128 xf86DrvMsg(0, X_INFO, "Trumpion block success\n");
9129#endif
9130 return TRUE;
9131}
9132#endif
9133
9134/* The Chrontel 700x is connected to the 630/730 via
9135 * the 630/730's DDC/I2C port.
9136 *
9137 * On 630(S)T chipset, the index changed from 0x11 to
9138 * 0x0a, possibly for working around the DDC problems
9139 */
9140
9141static BOOLEAN
9142SiS_SetChReg(SiS_Private *SiS_Pr, USHORT tempbx, USHORT myor)
9143{
9144 USHORT tempah,temp,i;
9145
9146 for(i=0; i<20; i++) { /* Do 20 attempts to write */
9147 if(i) {
9148 SiS_SetStop(SiS_Pr);
9149 SiS_DDC2Delay(SiS_Pr,SiS_I2CDELAYSHORT);
9150 }
9151 if(SiS_SetStart(SiS_Pr)) continue; /* Set start condition */
9152 tempah = SiS_Pr->SiS_DDC_DeviceAddr;
9153 temp = SiS_WriteDDC2Data(SiS_Pr,tempah); /* Write DAB (S0=0=write) */
9154 if(temp) continue; /* (ERROR: no ack) */
9155 tempah = tempbx & 0x00FF; /* Write RAB */
9156 tempah |= myor; /* (700x: set bit 7, see datasheet) */
9157 temp = SiS_WriteDDC2Data(SiS_Pr,tempah);
9158 if(temp) continue; /* (ERROR: no ack) */
9159 tempah = (tempbx & 0xFF00) >> 8;
9160 temp = SiS_WriteDDC2Data(SiS_Pr,tempah); /* Write data */
9161 if(temp) continue; /* (ERROR: no ack) */
9162 if(SiS_SetStop(SiS_Pr)) continue; /* Set stop condition */
9163 SiS_Pr->SiS_ChrontelInit = 1;
9164 return TRUE;
9165 }
9166 return FALSE;
9167}
9168
9169#if 0
9170#ifdef SIS300
9171/* Write Trumpion register */
9172static void
9173SiS_SetTrumpReg(SiS_Private *SiS_Pr, USHORT tempbx)
9174{
9175 SiS_Pr->SiS_DDC_DeviceAddr = 0xF0; /* DAB (Device Address Byte) */
9176 SiS_Pr->SiS_DDC_Index = 0x11; /* Bit 0 = SC; Bit 1 = SD */
9177 SiS_Pr->SiS_DDC_Data = 0x02; /* Bitmask in IndexReg for Data */
9178 SiS_Pr->SiS_DDC_Clk = 0x01; /* Bitmask in IndexReg for Clk */
9179 SiS_SetupDDCN(SiS_Pr);
9180 SiS_SetChReg(SiS_Pr, tempbx, 0);
9181}
9182#endif
9183#endif
9184
9185/* Write to Chrontel 700x */
9186/* Parameter is [Data (S15-S8) | Register no (S7-S0)] */
9187void
9188SiS_SetCH700x(SiS_Private *SiS_Pr, USHORT tempbx)
9189{
9190 SiS_Pr->SiS_DDC_DeviceAddr = 0xEA; /* DAB (Device Address Byte) */
9191
9192 if(!(SiS_Pr->SiS_ChrontelInit)) {
9193 SiS_Pr->SiS_DDC_Index = 0x11; /* Bit 0 = SC; Bit 1 = SD */
9194 SiS_Pr->SiS_DDC_Data = 0x02; /* Bitmask in IndexReg for Data */
9195 SiS_Pr->SiS_DDC_Clk = 0x01; /* Bitmask in IndexReg for Clk */
9196 SiS_SetupDDCN(SiS_Pr);
9197 }
9198
9199 if( (!(SiS_SetChReg(SiS_Pr, tempbx, 0x80))) &&
9200 (!(SiS_Pr->SiS_ChrontelInit)) ) {
9201 SiS_Pr->SiS_DDC_Index = 0x0a; /* Bit 7 = SC; Bit 6 = SD */
9202 SiS_Pr->SiS_DDC_Data = 0x80; /* Bitmask in IndexReg for Data */
9203 SiS_Pr->SiS_DDC_Clk = 0x40; /* Bitmask in IndexReg for Clk */
9204 SiS_SetupDDCN(SiS_Pr);
9205
9206 SiS_SetChReg(SiS_Pr, tempbx, 0x80);
9207 }
9208}
9209
9210/* Write to Chrontel 701x */
9211/* Parameter is [Data (S15-S8) | Register no (S7-S0)] */
9212void
9213SiS_SetCH701x(SiS_Private *SiS_Pr, USHORT tempbx)
9214{
9215 SiS_Pr->SiS_DDC_Index = 0x11; /* Bit 0 = SC; Bit 1 = SD */
9216 SiS_Pr->SiS_DDC_Data = 0x08; /* Bitmask in IndexReg for Data */
9217 SiS_Pr->SiS_DDC_Clk = 0x04; /* Bitmask in IndexReg for Clk */
9218 SiS_SetupDDCN(SiS_Pr);
9219 SiS_Pr->SiS_DDC_DeviceAddr = 0xEA; /* DAB (Device Address Byte) */
9220 SiS_SetChReg(SiS_Pr, tempbx, 0);
9221}
9222
Adrian Bunk75c96f82005-05-05 16:16:09 -07009223static void
Linus Torvalds1da177e2005-04-16 15:20:36 -07009224SiS_SetCH70xx(SiS_Private *SiS_Pr, USHORT tempbx)
9225{
9226 if(SiS_Pr->SiS_IF_DEF_CH70xx == 1)
9227 SiS_SetCH700x(SiS_Pr,tempbx);
9228 else
9229 SiS_SetCH701x(SiS_Pr,tempbx);
9230}
9231
9232static USHORT
9233SiS_GetChReg(SiS_Private *SiS_Pr, USHORT myor)
9234{
9235 USHORT tempah,temp,i;
9236
9237 for(i=0; i<20; i++) { /* Do 20 attempts to read */
9238 if(i) {
9239 SiS_SetStop(SiS_Pr);
9240 SiS_DDC2Delay(SiS_Pr,SiS_I2CDELAYSHORT);
9241 }
9242 if(SiS_SetStart(SiS_Pr)) continue; /* Set start condition */
9243 tempah = SiS_Pr->SiS_DDC_DeviceAddr;
9244 temp = SiS_WriteDDC2Data(SiS_Pr,tempah); /* Write DAB (S0=0=write) */
9245 if(temp) continue; /* (ERROR: no ack) */
9246 tempah = SiS_Pr->SiS_DDC_ReadAddr | myor; /* Write RAB (700x: | 0x80) */
9247 temp = SiS_WriteDDC2Data(SiS_Pr,tempah);
9248 if(temp) continue; /* (ERROR: no ack) */
9249 if (SiS_SetStart(SiS_Pr)) continue; /* Re-start */
9250 tempah = SiS_Pr->SiS_DDC_DeviceAddr | 0x01;/* DAB | 0x01 = Read */
9251 temp = SiS_WriteDDC2Data(SiS_Pr,tempah); /* DAB (S0=1=read) */
9252 if(temp) continue; /* (ERROR: no ack) */
9253 tempah = SiS_ReadDDC2Data(SiS_Pr,tempah); /* Read byte */
9254 if(SiS_SetStop(SiS_Pr)) continue; /* Stop condition */
9255 SiS_Pr->SiS_ChrontelInit = 1;
9256 return(tempah);
9257 }
9258 return 0xFFFF;
9259}
9260
9261#if 0
9262#ifdef SIS300
9263/* Read from Trumpion */
9264static USHORT
9265SiS_GetTrumpReg(SiS_Private *SiS_Pr, USHORT tempbx)
9266{
9267 SiS_Pr->SiS_DDC_DeviceAddr = 0xF0; /* DAB */
9268 SiS_Pr->SiS_DDC_Index = 0x11; /* Bit 0 = SC; Bit 1 = SD */
9269 SiS_Pr->SiS_DDC_Data = 0x02; /* Bitmask in IndexReg for Data */
9270 SiS_Pr->SiS_DDC_Clk = 0x01; /* Bitmask in IndexReg for Clk */
9271 SiS_SetupDDCN(SiS_Pr);
9272 SiS_Pr->SiS_DDC_ReadAddr = tempbx;
9273 return(SiS_GetChReg(SiS_Pr,0));
9274}
9275#endif
9276#endif
9277
9278/* Read from Chrontel 700x */
9279/* Parameter is [Register no (S7-S0)] */
9280USHORT
9281SiS_GetCH700x(SiS_Private *SiS_Pr, USHORT tempbx)
9282{
9283 USHORT result;
9284
9285 SiS_Pr->SiS_DDC_DeviceAddr = 0xEA; /* DAB */
9286
9287 if(!(SiS_Pr->SiS_ChrontelInit)) {
9288 SiS_Pr->SiS_DDC_Index = 0x11; /* Bit 0 = SC; Bit 1 = SD */
9289 SiS_Pr->SiS_DDC_Data = 0x02; /* Bitmask in IndexReg for Data */
9290 SiS_Pr->SiS_DDC_Clk = 0x01; /* Bitmask in IndexReg for Clk */
9291 SiS_SetupDDCN(SiS_Pr);
9292 }
9293
9294 SiS_Pr->SiS_DDC_ReadAddr = tempbx;
9295
9296 if( ((result = SiS_GetChReg(SiS_Pr,0x80)) == 0xFFFF) &&
9297 (!SiS_Pr->SiS_ChrontelInit) ) {
9298
9299 SiS_Pr->SiS_DDC_Index = 0x0a;
9300 SiS_Pr->SiS_DDC_Data = 0x80;
9301 SiS_Pr->SiS_DDC_Clk = 0x40;
9302 SiS_SetupDDCN(SiS_Pr);
9303
9304 result = SiS_GetChReg(SiS_Pr,0x80);
9305 }
9306 return(result);
9307}
9308
9309/* Read from Chrontel 701x */
9310/* Parameter is [Register no (S7-S0)] */
9311USHORT
9312SiS_GetCH701x(SiS_Private *SiS_Pr, USHORT tempbx)
9313{
9314 SiS_Pr->SiS_DDC_Index = 0x11; /* Bit 0 = SC; Bit 1 = SD */
9315 SiS_Pr->SiS_DDC_Data = 0x08; /* Bitmask in IndexReg for Data */
9316 SiS_Pr->SiS_DDC_Clk = 0x04; /* Bitmask in IndexReg for Clk */
9317 SiS_SetupDDCN(SiS_Pr);
9318 SiS_Pr->SiS_DDC_DeviceAddr = 0xEA; /* DAB */
9319
9320 SiS_Pr->SiS_DDC_ReadAddr = tempbx;
9321
9322 return(SiS_GetChReg(SiS_Pr,0));
9323}
9324
9325/* Read from Chrontel 70xx */
9326/* Parameter is [Register no (S7-S0)] */
Adrian Bunk75c96f82005-05-05 16:16:09 -07009327static USHORT
Linus Torvalds1da177e2005-04-16 15:20:36 -07009328SiS_GetCH70xx(SiS_Private *SiS_Pr, USHORT tempbx)
9329{
9330 if(SiS_Pr->SiS_IF_DEF_CH70xx == 1)
9331 return(SiS_GetCH700x(SiS_Pr, tempbx));
9332 else
9333 return(SiS_GetCH701x(SiS_Pr, tempbx));
9334}
9335
9336/* Our own DDC functions */
9337static USHORT
9338SiS_InitDDCRegs(SiS_Private *SiS_Pr, unsigned long VBFlags, int VGAEngine,
9339 USHORT adaptnum, USHORT DDCdatatype, BOOLEAN checkcr32)
9340{
9341 unsigned char ddcdtype[] = { 0xa0, 0xa0, 0xa0, 0xa2, 0xa6 };
9342 unsigned char flag, cr32;
9343 USHORT temp = 0, myadaptnum = adaptnum;
9344
9345 if(adaptnum != 0) {
9346 if(!(VBFlags & (VB_301|VB_301B|VB_301C|VB_302B))) return 0xFFFF;
9347 if((VBFlags & VB_30xBDH) && (adaptnum == 1)) return 0xFFFF;
9348 }
9349
9350 /* adapternum for SiS bridges: 0 = CRT1, 1 = LCD, 2 = VGA2 */
9351
9352 SiS_Pr->SiS_ChrontelInit = 0; /* force re-detection! */
9353
9354 SiS_Pr->SiS_DDC_SecAddr = 0;
9355 SiS_Pr->SiS_DDC_DeviceAddr = ddcdtype[DDCdatatype];
9356 SiS_Pr->SiS_DDC_Port = SiS_Pr->SiS_P3c4;
9357 SiS_Pr->SiS_DDC_Index = 0x11;
9358 flag = 0xff;
9359
9360 cr32 = SiS_GetReg(SiS_Pr->SiS_P3d4,0x32);
9361
9362#if 0
9363 if(VBFlags & VB_SISBRIDGE) {
9364 if(myadaptnum == 0) {
9365 if(!(cr32 & 0x20)) {
9366 myadaptnum = 2;
9367 if(!(cr32 & 0x10)) {
9368 myadaptnum = 1;
9369 if(!(cr32 & 0x08)) {
9370 myadaptnum = 0;
9371 }
9372 }
9373 }
9374 }
9375 }
9376#endif
9377
9378 if(VGAEngine == SIS_300_VGA) { /* 300 series */
9379
9380 if(myadaptnum != 0) {
9381 flag = 0;
9382 if(VBFlags & VB_SISBRIDGE) {
9383 SiS_Pr->SiS_DDC_Port = SiS_Pr->SiS_Part4Port;
9384 SiS_Pr->SiS_DDC_Index = 0x0f;
9385 }
9386 }
9387
9388 if(!(VBFlags & VB_301)) {
9389 if((cr32 & 0x80) && (checkcr32)) {
9390 if(myadaptnum >= 1) {
9391 if(!(cr32 & 0x08)) {
9392 myadaptnum = 1;
9393 if(!(cr32 & 0x10)) return 0xFFFF;
9394 }
9395 }
9396 }
9397 }
9398
9399 temp = 4 - (myadaptnum * 2);
9400 if(flag) temp = 0;
9401
9402 } else { /* 315/330 series */
9403
9404 /* here we simplify: 0 = CRT1, 1 = CRT2 (VGA, LCD) */
9405
9406 if(VBFlags & VB_SISBRIDGE) {
9407 if(myadaptnum == 2) {
9408 myadaptnum = 1;
9409 }
9410 }
9411
9412 if(myadaptnum == 1) {
9413 flag = 0;
9414 if(VBFlags & VB_SISBRIDGE) {
9415 SiS_Pr->SiS_DDC_Port = SiS_Pr->SiS_Part4Port;
9416 SiS_Pr->SiS_DDC_Index = 0x0f;
9417 }
9418 }
9419
9420 if((cr32 & 0x80) && (checkcr32)) {
9421 if(myadaptnum >= 1) {
9422 if(!(cr32 & 0x08)) {
9423 myadaptnum = 1;
9424 if(!(cr32 & 0x10)) return 0xFFFF;
9425 }
9426 }
9427 }
9428
9429 temp = myadaptnum;
9430 if(myadaptnum == 1) {
9431 temp = 0;
9432 if(VBFlags & VB_LVDS) flag = 0xff;
9433 }
9434
9435 if(flag) temp = 0;
9436 }
9437
9438 SiS_Pr->SiS_DDC_Data = 0x02 << temp;
9439 SiS_Pr->SiS_DDC_Clk = 0x01 << temp;
9440
9441 SiS_SetupDDCN(SiS_Pr);
9442
9443#ifdef TWDEBUG
9444 xf86DrvMsg(0, X_INFO, "DDC Port %x Index %x Shift %d\n",
9445 SiS_Pr->SiS_DDC_Port, SiS_Pr->SiS_DDC_Index, temp);
9446#endif
9447
9448 return 0;
9449}
9450
9451static USHORT
9452SiS_WriteDABDDC(SiS_Private *SiS_Pr)
9453{
9454 if(SiS_SetStart(SiS_Pr)) return 0xFFFF;
9455 if(SiS_WriteDDC2Data(SiS_Pr, SiS_Pr->SiS_DDC_DeviceAddr)) {
9456 return 0xFFFF;
9457 }
9458 if(SiS_WriteDDC2Data(SiS_Pr, SiS_Pr->SiS_DDC_SecAddr)) {
9459 return 0xFFFF;
9460 }
9461 return(0);
9462}
9463
9464static USHORT
9465SiS_PrepareReadDDC(SiS_Private *SiS_Pr)
9466{
9467 if(SiS_SetStart(SiS_Pr)) return 0xFFFF;
9468 if(SiS_WriteDDC2Data(SiS_Pr, (SiS_Pr->SiS_DDC_DeviceAddr | 0x01))) {
9469 return 0xFFFF;
9470 }
9471 return(0);
9472}
9473
9474static USHORT
9475SiS_PrepareDDC(SiS_Private *SiS_Pr)
9476{
9477 if(SiS_WriteDABDDC(SiS_Pr)) SiS_WriteDABDDC(SiS_Pr);
9478 if(SiS_PrepareReadDDC(SiS_Pr)) return(SiS_PrepareReadDDC(SiS_Pr));
9479 return(0);
9480}
9481
9482static void
9483SiS_SendACK(SiS_Private *SiS_Pr, USHORT yesno)
9484{
9485 SiS_SetSCLKLow(SiS_Pr);
9486 if(yesno) {
9487 SiS_SetRegANDOR(SiS_Pr->SiS_DDC_Port,
9488 SiS_Pr->SiS_DDC_Index,
9489 SiS_Pr->SiS_DDC_NData,
9490 SiS_Pr->SiS_DDC_Data);
9491 } else {
9492 SiS_SetRegANDOR(SiS_Pr->SiS_DDC_Port,
9493 SiS_Pr->SiS_DDC_Index,
9494 SiS_Pr->SiS_DDC_NData,
9495 0);
9496 }
9497 SiS_SetSCLKHigh(SiS_Pr);
9498}
9499
9500static USHORT
9501SiS_DoProbeDDC(SiS_Private *SiS_Pr)
9502{
9503 unsigned char mask, value;
9504 USHORT temp, ret=0;
9505 BOOLEAN failed = FALSE;
9506
9507 SiS_SetSwitchDDC2(SiS_Pr);
9508 if(SiS_PrepareDDC(SiS_Pr)) {
9509 SiS_SetStop(SiS_Pr);
9510#ifdef TWDEBUG
9511 xf86DrvMsg(0, X_INFO, "Probe: Prepare failed\n");
9512#endif
9513 return(0xFFFF);
9514 }
9515 mask = 0xf0;
9516 value = 0x20;
9517 if(SiS_Pr->SiS_DDC_DeviceAddr == 0xa0) {
9518 temp = (unsigned char)SiS_ReadDDC2Data(SiS_Pr, 0);
9519 SiS_SendACK(SiS_Pr, 0);
9520 if(temp == 0) {
9521 mask = 0xff;
9522 value = 0xff;
9523 } else {
9524 failed = TRUE;
9525 ret = 0xFFFF;
9526#ifdef TWDEBUG
9527 xf86DrvMsg(0, X_INFO, "Probe: Read 1 failed\n");
9528#endif
9529 }
9530 }
9531 if(failed == FALSE) {
9532 temp = (unsigned char)SiS_ReadDDC2Data(SiS_Pr, 0);
9533 SiS_SendACK(SiS_Pr, 1);
9534 temp &= mask;
9535 if(temp == value) ret = 0;
9536 else {
9537 ret = 0xFFFF;
9538#ifdef TWDEBUG
9539 xf86DrvMsg(0, X_INFO, "Probe: Read 2 failed\n");
9540#endif
9541 if(SiS_Pr->SiS_DDC_DeviceAddr == 0xa0) {
9542 if(temp == 0x30) ret = 0;
9543 }
9544 }
9545 }
9546 SiS_SetStop(SiS_Pr);
9547 return(ret);
9548}
9549
9550static USHORT
9551SiS_ProbeDDC(SiS_Private *SiS_Pr)
9552{
9553 USHORT flag;
9554
9555 flag = 0x180;
9556 SiS_Pr->SiS_DDC_DeviceAddr = 0xa0;
9557 if(!(SiS_DoProbeDDC(SiS_Pr))) flag |= 0x02;
9558 SiS_Pr->SiS_DDC_DeviceAddr = 0xa2;
9559 if(!(SiS_DoProbeDDC(SiS_Pr))) flag |= 0x08;
9560 SiS_Pr->SiS_DDC_DeviceAddr = 0xa6;
9561 if(!(SiS_DoProbeDDC(SiS_Pr))) flag |= 0x10;
9562 if(!(flag & 0x1a)) flag = 0;
9563 return(flag);
9564}
9565
9566static USHORT
9567SiS_ReadDDC(SiS_Private *SiS_Pr, USHORT DDCdatatype, unsigned char *buffer)
9568{
9569 USHORT flag, length, i;
9570 unsigned char chksum,gotcha;
9571
9572 if(DDCdatatype > 4) return 0xFFFF;
9573
9574 flag = 0;
9575 SiS_SetSwitchDDC2(SiS_Pr);
9576 if(!(SiS_PrepareDDC(SiS_Pr))) {
9577 length = 127;
9578 if(DDCdatatype != 1) length = 255;
9579 chksum = 0;
9580 gotcha = 0;
9581 for(i=0; i<length; i++) {
9582 buffer[i] = (unsigned char)SiS_ReadDDC2Data(SiS_Pr, 0);
9583 chksum += buffer[i];
9584 gotcha |= buffer[i];
9585 SiS_SendACK(SiS_Pr, 0);
9586 }
9587 buffer[i] = (unsigned char)SiS_ReadDDC2Data(SiS_Pr, 0);
9588 chksum += buffer[i];
9589 SiS_SendACK(SiS_Pr, 1);
9590 if(gotcha) flag = (USHORT)chksum;
9591 else flag = 0xFFFF;
9592 } else {
9593 flag = 0xFFFF;
9594 }
9595 SiS_SetStop(SiS_Pr);
9596 return(flag);
9597}
9598
9599/* Our private DDC functions
9600
9601 It complies somewhat with the corresponding VESA function
9602 in arguments and return values.
9603
9604 Since this is probably called before the mode is changed,
9605 we use our pre-detected pSiS-values instead of SiS_Pr as
9606 regards chipset and video bridge type.
9607
9608 Arguments:
9609 adaptnum: 0=CRT1(analog), 1=CRT2/LCD(digital), 2=CRT2/VGA2(analog)
9610 CRT2 DDC is only supported on SiS301, 301B, 301C, 302B.
9611 LCDA is CRT1, but DDC is read from CRT2 port.
9612 DDCdatatype: 0=Probe, 1=EDID, 2=EDID+VDIF, 3=EDID V2 (P&D), 4=EDID V2 (FPDI-2)
9613 buffer: ptr to 256 data bytes which will be filled with read data.
9614
9615 Returns 0xFFFF if error, otherwise
9616 if DDCdatatype > 0: Returns 0 if reading OK (included a correct checksum)
9617 if DDCdatatype = 0: Returns supported DDC modes
9618
9619 */
9620USHORT
9621SiS_HandleDDC(SiS_Private *SiS_Pr, unsigned long VBFlags, int VGAEngine,
9622 USHORT adaptnum, USHORT DDCdatatype, unsigned char *buffer)
9623{
9624 unsigned char sr1f,cr17=1;
9625 USHORT result;
9626
9627 if(adaptnum > 2) return 0xFFFF;
9628 if(DDCdatatype > 4) return 0xFFFF;
9629 if((!(VBFlags & VB_VIDEOBRIDGE)) && (adaptnum > 0)) return 0xFFFF;
9630 if(SiS_InitDDCRegs(SiS_Pr, VBFlags, VGAEngine, adaptnum, DDCdatatype, FALSE) == 0xFFFF) return 0xFFFF;
9631
9632 sr1f = SiS_GetReg(SiS_Pr->SiS_P3c4,0x1f);
9633 SiS_SetRegANDOR(SiS_Pr->SiS_P3c4,0x1f,0x3f,0x04);
9634 if(VGAEngine == SIS_300_VGA) {
9635 cr17 = SiS_GetReg(SiS_Pr->SiS_P3d4,0x17) & 0x80;
9636 if(!cr17) {
9637 SiS_SetRegOR(SiS_Pr->SiS_P3d4,0x17,0x80);
9638 SiS_SetReg(SiS_Pr->SiS_P3c4,0x00,0x01);
9639 SiS_SetReg(SiS_Pr->SiS_P3c4,0x00,0x03);
9640 }
9641 }
9642 if((sr1f) || (!cr17)) {
9643 SiS_WaitRetrace1(SiS_Pr);
9644 SiS_WaitRetrace1(SiS_Pr);
9645 SiS_WaitRetrace1(SiS_Pr);
9646 SiS_WaitRetrace1(SiS_Pr);
9647 }
9648
9649 if(DDCdatatype == 0) {
9650 result = SiS_ProbeDDC(SiS_Pr);
9651 } else {
9652 result = SiS_ReadDDC(SiS_Pr, DDCdatatype, buffer);
9653 if((!result) && (DDCdatatype == 1)) {
9654 if((buffer[0] == 0x00) && (buffer[1] == 0xff) &&
9655 (buffer[2] == 0xff) && (buffer[3] == 0xff) &&
9656 (buffer[4] == 0xff) && (buffer[5] == 0xff) &&
9657 (buffer[6] == 0xff) && (buffer[7] == 0x00) &&
9658 (buffer[0x12] == 1)) {
9659 if(adaptnum == 1) {
9660 if(!(buffer[0x14] & 0x80)) result = 0xFFFE;
9661 } else {
9662 if(buffer[0x14] & 0x80) result = 0xFFFE;
9663 }
9664 }
9665 }
9666 }
9667 SiS_SetReg(SiS_Pr->SiS_P3c4,0x1f,sr1f);
9668 if(VGAEngine == SIS_300_VGA) {
9669 SiS_SetRegANDOR(SiS_Pr->SiS_P3d4,0x17,0x7f,cr17);
9670 }
9671 return result;
9672}
9673
9674#ifdef LINUX_XF86
9675
9676static BOOLEAN
9677checkedid1(unsigned char *buffer)
9678{
9679 /* Check header */
9680 if((buffer[0] != 0x00) ||
9681 (buffer[1] != 0xff) ||
9682 (buffer[2] != 0xff) ||
9683 (buffer[3] != 0xff) ||
9684 (buffer[4] != 0xff) ||
9685 (buffer[5] != 0xff) ||
9686 (buffer[6] != 0xff) ||
9687 (buffer[7] != 0x00))
9688 return FALSE;
9689
9690 /* Check EDID version and revision */
9691 if((buffer[0x12] != 1) || (buffer[0x13] > 4)) return FALSE;
9692
9693 /* Check week of manufacture for sanity */
9694 if(buffer[0x10] > 53) return FALSE;
9695
9696 /* Check year of manufacture for sanity */
9697 if(buffer[0x11] > 40) return FALSE;
9698
9699 return TRUE;
9700}
9701
9702static BOOLEAN
9703checkedid2(unsigned char *buffer)
9704{
9705 USHORT year = buffer[6] | (buffer[7] << 8);
9706
9707 /* Check EDID version */
9708 if((buffer[0] & 0xf0) != 0x20) return FALSE;
9709
9710 /* Check week of manufacture for sanity */
9711 if(buffer[5] > 53) return FALSE;
9712
9713 /* Check year of manufacture for sanity */
9714 if((year != 0) && ((year < 1990) || (year > 2030))) return FALSE;
9715
9716 return TRUE;
9717}
9718
9719/* Sense the LCD parameters (CR36, CR37) via DDC */
9720/* SiS30x(B) only */
9721USHORT
9722SiS_SenseLCDDDC(SiS_Private *SiS_Pr, SISPtr pSiS)
9723{
9724 USHORT DDCdatatype, paneltype, flag, xres=0, yres=0;
9725 USHORT index, myindex, lumsize, numcodes, panelvendor, panelproduct;
9726 int maxx=0, maxy=0, prefx=0, prefy=0;
9727 unsigned char cr37=0, seekcode;
9728 BOOLEAN checkexpand = FALSE;
9729 BOOLEAN havesync = FALSE;
9730 BOOLEAN indb = FALSE;
9731 int retry, i;
9732 unsigned char buffer[256];
9733
9734 for(i=0; i<7; i++) SiS_Pr->CP_DataValid[i] = FALSE;
9735 SiS_Pr->CP_HaveCustomData = FALSE;
9736 SiS_Pr->CP_MaxX = SiS_Pr->CP_MaxY = SiS_Pr->CP_MaxClock = 0;
9737 SiS_Pr->CP_PreferredX = SiS_Pr->CP_PreferredY = 0;
9738 SiS_Pr->CP_PreferredIndex = -1;
9739 SiS_Pr->CP_PrefClock = 0;
9740 SiS_Pr->PanelSelfDetected = FALSE;
9741
9742 if(!(pSiS->VBFlags & (VB_301|VB_301B|VB_301C|VB_302B))) return 0;
9743 if(pSiS->VBFlags & VB_30xBDH) return 0;
9744
9745 if(SiS_InitDDCRegs(SiS_Pr, pSiS->VBFlags, pSiS->VGAEngine, 1, 0, FALSE) == 0xFFFF) return 0;
9746
9747 SiS_Pr->SiS_DDC_SecAddr = 0x00;
9748
9749 /* Probe supported DA's */
9750 flag = SiS_ProbeDDC(SiS_Pr);
9751#ifdef TWDEBUG
9752 xf86DrvMsg(pSiS->pScrn->scrnIndex, X_INFO,
9753 "CRT2 DDC capabilities 0x%x\n", flag);
9754#endif
9755 if(flag & 0x10) {
9756 SiS_Pr->SiS_DDC_DeviceAddr = 0xa6; /* EDID V2 (FP) */
9757 DDCdatatype = 4;
9758 } else if(flag & 0x08) {
9759 SiS_Pr->SiS_DDC_DeviceAddr = 0xa2; /* EDID V2 (P&D-D Monitor) */
9760 DDCdatatype = 3;
9761 } else if(flag & 0x02) {
9762 SiS_Pr->SiS_DDC_DeviceAddr = 0xa0; /* EDID V1 */
9763 DDCdatatype = 1;
9764 } else return 0; /* no DDC support (or no device attached) */
9765
9766 /* Read the entire EDID */
9767 retry = 2;
9768 do {
9769 if(SiS_ReadDDC(SiS_Pr, DDCdatatype, buffer)) {
9770 xf86DrvMsg(pSiS->pScrn->scrnIndex, X_PROBED,
9771 "CRT2: DDC read failed (attempt %d), %s\n",
9772 (3-retry), (retry == 1) ? "giving up" : "retrying");
9773 retry--;
9774 if(retry == 0) return 0xFFFF;
9775 } else break;
9776 } while(1);
9777
9778#ifdef TWDEBUG
9779 for(i=0; i<256; i+=16) {
9780 xf86DrvMsg(pSiS->pScrn->scrnIndex, X_PROBED,
9781 "%02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x\n",
9782 buffer[i], buffer[i+1], buffer[i+2], buffer[i+3],
9783 buffer[i+4], buffer[i+5], buffer[i+6], buffer[i+7],
9784 buffer[i+8], buffer[i+9], buffer[i+10], buffer[i+11],
9785 buffer[i+12], buffer[i+13], buffer[i+14], buffer[i+15]);
9786 }
9787#endif
9788
9789 /* Analyze EDID and retrieve LCD panel information */
9790 paneltype = 0;
9791 switch(DDCdatatype) {
9792 case 1: /* Analyze EDID V1 */
9793 /* Catch a few clear cases: */
9794 if(!(checkedid1(buffer))) {
9795 xf86DrvMsg(pSiS->pScrn->scrnIndex, X_PROBED,
9796 "LCD sense: EDID corrupt\n");
9797 return 0;
9798 }
9799
9800 if(!(buffer[0x14] & 0x80)) {
9801 xf86DrvMsg(pSiS->pScrn->scrnIndex, X_PROBED,
9802 "LCD sense: Attached display expects analog input (0x%02x)\n",
9803 buffer[0x14]);
9804 return 0;
9805 }
9806
9807 if((buffer[0x18] & 0x18) != 0x08) {
9808 xf86DrvMsg(pSiS->pScrn->scrnIndex, X_PROBED,
9809 "LCD sense: Warning: Attached display is not of RGB but of %s type (0x%02x)\n",
9810 ((buffer[0x18] & 0x18) == 0x00) ? "monochrome/greyscale" :
9811 ( ((buffer[0x18] & 0x18) == 0x10) ? "non-RGB multicolor" :
9812 "undefined"),
9813 buffer[0x18]);
9814 }
9815
9816 /* Now analyze the first Detailed Timing Block and see
9817 * if the preferred timing mode is stored there. If so,
9818 * check if this is a standard panel for which we already
9819 * know the timing.
9820 */
9821
9822 paneltype = Panel_Custom;
9823 checkexpand = FALSE;
9824
9825 panelvendor = buffer[9] | (buffer[8] << 8);
9826 panelproduct = buffer[10] | (buffer[11] << 8);
9827
9828 /* Overrule bogus preferred modes from database */
9829 if((indb = SiS_FindPanelFromDB(pSiS, panelvendor, panelproduct, &maxx, &maxy, &prefx, &prefy))) {
9830 if(prefx) SiS_Pr->CP_PreferredX = xres = prefx;
9831 if(prefy) SiS_Pr->CP_PreferredY = yres = prefy;
9832 }
9833
9834 if(buffer[0x18] & 0x02) {
9835
9836 USHORT pclk = (buffer[0x36] | (buffer[0x37] << 8));
9837 USHORT phb = (buffer[0x39] | ((buffer[0x3a] & 0x0f) << 8));
9838 USHORT pvb = (buffer[0x3c] | ((buffer[0x3d] & 0x0f) << 8));
9839
9840 if(!xres) SiS_Pr->CP_PreferredX = xres = buffer[0x38] | ((buffer[0x3a] & 0xf0) << 4);
9841 if(!yres) SiS_Pr->CP_PreferredY = yres = buffer[0x3b] | ((buffer[0x3d] & 0xf0) << 4);
9842
9843 switch(xres) {
9844#if 0 /* Treat as custom */
9845 case 800:
9846 if(yres == 600) {
9847 paneltype = Panel_800x600;
9848 checkexpand = TRUE;
9849 }
9850 break;
9851#endif
9852 case 1024:
9853 if(yres == 768) {
9854 paneltype = Panel_1024x768;
9855 checkexpand = TRUE;
9856 }
9857 break;
9858 case 1280:
9859 if(yres == 1024) {
9860 paneltype = Panel_1280x1024;
9861 checkexpand = TRUE;
9862 } else if(yres == 960) {
9863 if(pSiS->VGAEngine == SIS_300_VGA) {
9864 paneltype = Panel300_1280x960;
9865 } else {
9866 paneltype = Panel310_1280x960;
9867 }
9868 } else if(yres == 768) {
9869 if( (pclk == 8100) &&
9870 (phb == (1688 - 1280)) &&
9871 (pvb == (802 - 768)) ) {
9872 paneltype = Panel_1280x768;
9873 checkexpand = FALSE;
9874 cr37 |= 0x10;
9875 }
9876 } else if(yres == 800) {
9877 if( (pclk == 6900) &&
9878 (phb == (1408 - 1280)) &&
9879 (pvb == (816 - 800)) ) {
9880 paneltype = Panel_1280x800;
9881 }
9882 }
9883 break;
9884 case 1400:
9885 if(pSiS->VGAEngine == SIS_315_VGA) {
9886 if(yres == 1050) {
9887 paneltype = Panel310_1400x1050;
9888 checkexpand = TRUE;
9889 }
9890 }
9891 break;
9892 case 1600:
9893 if(pSiS->VGAEngine == SIS_315_VGA) {
9894 if(pSiS->VBFlags & VB_301C) {
9895 if(yres == 1200) {
9896 paneltype = Panel310_1600x1200;
9897 checkexpand = TRUE;
9898 }
9899 }
9900 }
9901 break;
9902 }
9903
9904 /* Save sync: This is used if "Pass 1:1" is off; in this case
9905 * we always use the panel's native mode = this "preferred mode"
9906 * we just have been analysing. Hence, we also need its sync.
9907 */
9908 if((buffer[0x47] & 0x18) == 0x18) {
9909 cr37 |= ((((buffer[0x47] & 0x06) ^ 0x06) << 5) | 0x20);
9910 havesync = TRUE;
9911 } else {
9912 /* What now? There is no digital separate output timing... */
9913 xf86DrvMsg(pSiS->pScrn->scrnIndex, X_WARNING,
9914 "LCD sense: Unable to retrieve Sync polarity information\n");
9915 cr37 |= 0xc0; /* Default */
9916 }
9917
9918 }
9919
9920 /* Check against our database; eg. Sanyo Z2 projector reports
9921 * 1024x768 as preferred mode, although it supports 1280x720
9922 * natively in non-HDCP mode. Treat such wrongly reporting
9923 * panels as custom and fixup actual maximum resolutions.
9924 */
9925 if(paneltype != Panel_Custom) {
9926 if(indb) {
9927 paneltype = Panel_Custom;
9928 SiS_Pr->CP_MaxX = maxx;
9929 SiS_Pr->CP_MaxY = maxy;
9930 /* Leave preferred unchanged (MUST contain a valid mode!) */
9931 }
9932 }
9933
9934 /* If we still don't know what panel this is, we take it
9935 * as a custom panel and derive the timing data from the
9936 * detailed timing blocks
9937 */
9938 if(paneltype == Panel_Custom) {
9939
9940 int i, temp, base = 0x36;
9941 unsigned long estpack;
9942 const unsigned short estx[] = {
9943 720, 720, 640, 640, 640, 640, 800, 800,
9944 800, 800, 832,1024,1024,1024,1024,1280,
9945 1152
9946 };
9947 const unsigned short esty[] = {
9948 400, 400, 480, 480, 480, 480, 600, 600,
9949 600, 600, 624, 768, 768, 768, 768,1024,
9950 870
9951 };
9952 const int estclk[] = {
9953 0, 0, 25100, 0, 31500, 31500, 36100, 40000,
9954 50100, 49500, 0, 0, 65100, 75200, 78700,135200,
9955 0
9956 };
9957
9958 paneltype = 0;
9959 SiS_Pr->CP_Supports64048075 = TRUE;
9960
9961 /* Find the maximum resolution */
9962
9963 /* 1. From Established timings */
9964 estpack = (buffer[0x23] << 9) | (buffer[0x24] << 1) | ((buffer[0x25] >> 7) & 0x01);
9965 for(i=16; i>=0; i--) {
9966 if(estpack & (1 << i)) {
9967 if(estx[16 - i] > SiS_Pr->CP_MaxX) SiS_Pr->CP_MaxX = estx[16 - i];
9968 if(esty[16 - i] > SiS_Pr->CP_MaxY) SiS_Pr->CP_MaxY = esty[16 - i];
9969 if(estclk[16 - i] > SiS_Pr->CP_MaxClock) SiS_Pr->CP_MaxClock = estclk[16 - i];
9970 }
9971 }
9972
9973 /* By default we drive the LCD at 75Hz in 640x480 mode; if
9974 * the panel does not provide this mode, use 60hz
9975 */
9976 if(!(buffer[0x23] & 0x04)) SiS_Pr->CP_Supports64048075 = FALSE;
9977
9978 /* 2. From Standard Timings */
9979 for(i=0x26; i < 0x36; i+=2) {
9980 if((buffer[i] != 0x01) && (buffer[i+1] != 0x01)) {
9981 temp = (buffer[i] + 31) * 8;
9982 if(temp > SiS_Pr->CP_MaxX) SiS_Pr->CP_MaxX = temp;
9983 switch((buffer[i+1] & 0xc0) >> 6) {
9984 case 0x03: temp = temp * 9 / 16; break;
9985 case 0x02: temp = temp * 4 / 5; break;
9986 case 0x01: temp = temp * 3 / 4; break;
9987 }
9988 if(temp > SiS_Pr->CP_MaxY) SiS_Pr->CP_MaxY = temp;
9989 }
9990 }
9991
9992 /* Now extract the Detailed Timings and convert them into modes */
9993
9994 for(i = 0; i < 4; i++, base += 18) {
9995
9996 /* Is this a detailed timing block or a monitor descriptor? */
9997 if(buffer[base] || buffer[base+1] || buffer[base+2]) {
9998
9999 xres = buffer[base+2] | ((buffer[base+4] & 0xf0) << 4);
10000 yres = buffer[base+5] | ((buffer[base+7] & 0xf0) << 4);
10001
10002 SiS_Pr->CP_HDisplay[i] = xres;
10003 SiS_Pr->CP_HSyncStart[i] = xres + (buffer[base+8] | ((buffer[base+11] & 0xc0) << 2));
10004 SiS_Pr->CP_HSyncEnd[i] = SiS_Pr->CP_HSyncStart[i] + (buffer[base+9] | ((buffer[base+11] & 0x30) << 4));
10005 SiS_Pr->CP_HTotal[i] = xres + (buffer[base+3] | ((buffer[base+4] & 0x0f) << 8));
10006 SiS_Pr->CP_HBlankStart[i] = xres + 1;
10007 SiS_Pr->CP_HBlankEnd[i] = SiS_Pr->CP_HTotal[i];
10008
10009 SiS_Pr->CP_VDisplay[i] = yres;
10010 SiS_Pr->CP_VSyncStart[i] = yres + (((buffer[base+10] & 0xf0) >> 4) | ((buffer[base+11] & 0x0c) << 2));
10011 SiS_Pr->CP_VSyncEnd[i] = SiS_Pr->CP_VSyncStart[i] + ((buffer[base+10] & 0x0f) | ((buffer[base+11] & 0x03) << 4));
10012 SiS_Pr->CP_VTotal[i] = yres + (buffer[base+6] | ((buffer[base+7] & 0x0f) << 8));
10013 SiS_Pr->CP_VBlankStart[i] = yres + 1;
10014 SiS_Pr->CP_VBlankEnd[i] = SiS_Pr->CP_VTotal[i];
10015
10016 SiS_Pr->CP_Clock[i] = (buffer[base] | (buffer[base+1] << 8)) * 10;
10017
10018 SiS_Pr->CP_DataValid[i] = TRUE;
10019
10020 /* Sort out invalid timings, interlace and too high clocks */
10021 if((SiS_Pr->CP_HDisplay[i] & 7) ||
10022 (SiS_Pr->CP_HDisplay[i] > SiS_Pr->CP_HSyncStart[i]) ||
10023 (SiS_Pr->CP_HDisplay[i] >= SiS_Pr->CP_HSyncEnd[i]) ||
10024 (SiS_Pr->CP_HDisplay[i] >= SiS_Pr->CP_HTotal[i]) ||
10025 (SiS_Pr->CP_HSyncStart[i] >= SiS_Pr->CP_HSyncEnd[i]) ||
10026 (SiS_Pr->CP_HSyncStart[i] > SiS_Pr->CP_HTotal[i]) ||
10027 (SiS_Pr->CP_HSyncEnd[i] > SiS_Pr->CP_HTotal[i]) ||
10028 (SiS_Pr->CP_VDisplay[i] > SiS_Pr->CP_VSyncStart[i]) ||
10029 (SiS_Pr->CP_VDisplay[i] >= SiS_Pr->CP_VSyncEnd[i]) ||
10030 (SiS_Pr->CP_VDisplay[i] >= SiS_Pr->CP_VTotal[i]) ||
10031 (SiS_Pr->CP_VSyncStart[i] > SiS_Pr->CP_VSyncEnd[i]) ||
10032 (SiS_Pr->CP_VSyncStart[i] > SiS_Pr->CP_VTotal[i]) ||
10033 (SiS_Pr->CP_VSyncEnd[i] > SiS_Pr->CP_VTotal[i]) ||
10034 (((pSiS->VBFlags & VB_301C) && (SiS_Pr->CP_Clock[i] > 162500)) ||
10035 ((!(pSiS->VBFlags & VB_301C)) &&
10036 ((SiS_Pr->CP_Clock[i] > 108200) || (SiS_Pr->CP_VDisplay[i] > 1024) ||
10037 (SiS_Pr->CP_HDisplay[i] > 1600)))) ||
10038 (buffer[base+17] & 0x80)) {
10039
10040 SiS_Pr->CP_DataValid[i] = FALSE;
10041
10042 } else {
10043
10044 SiS_Pr->CP_HaveCustomData = TRUE;
10045
10046 if(xres > SiS_Pr->CP_MaxX) SiS_Pr->CP_MaxX = xres;
10047 if(yres > SiS_Pr->CP_MaxY) SiS_Pr->CP_MaxY = yres;
10048 if(SiS_Pr->CP_Clock[i] > SiS_Pr->CP_MaxClock) SiS_Pr->CP_MaxClock = SiS_Pr->CP_Clock[i];
10049
10050 if((SiS_Pr->CP_PreferredX == xres) && (SiS_Pr->CP_PreferredY == yres)) {
10051 SiS_Pr->CP_PreferredIndex = i;
10052 SiS_MakeClockRegs(pSiS->pScrn, SiS_Pr->CP_Clock[i], &SiS_Pr->CP_PrefSR2B, &SiS_Pr->CP_PrefSR2C);
10053 SiS_Pr->CP_PrefClock = (SiS_Pr->CP_Clock[i] / 1000) + 1;
10054 }
10055
10056 /* Extract the sync polarisation information. This only works
10057 * if the Flags indicate a digital separate output.
10058 */
10059 if((buffer[base+17] & 0x18) == 0x18) {
10060 SiS_Pr->CP_HSync_P[i] = (buffer[base+17] & 0x02) ? TRUE : FALSE;
10061 SiS_Pr->CP_VSync_P[i] = (buffer[base+17] & 0x04) ? TRUE : FALSE;
10062 SiS_Pr->CP_SyncValid[i] = TRUE;
10063 if((i == SiS_Pr->CP_PreferredIndex) && (!havesync)) {
10064 cr37 |= ((((buffer[base+17] & 0x06) ^ 0x06) << 5) | 0x20);
10065 havesync = TRUE;
10066 }
10067 } else {
10068 SiS_Pr->CP_SyncValid[i] = FALSE;
10069 }
10070
10071 }
10072
10073 } else if((!buffer[base]) && (!buffer[base+1]) && (!buffer[base+2]) && (!buffer[base+4])) {
10074
10075 /* Maximum pixclock from Monitor Range Limits */
10076 if((buffer[base+3] == 0xfd) && (buffer[base+9] != 0xff)) {
10077 int maxclk = buffer[base+9] * 10;
10078 /* More than 170 is not supported anyway */
10079 if(maxclk <= 170) SiS_Pr->CP_MaxClock = maxclk * 1000;
10080 }
10081
10082 }
10083
10084 }
10085
10086 if(SiS_Pr->CP_MaxX && SiS_Pr->CP_MaxY) {
10087 paneltype = Panel_Custom;
10088 checkexpand = FALSE;
10089 cr37 |= 0x10;
10090 SiS_Pr->CP_Vendor = panelvendor;
10091 SiS_Pr->CP_Product = panelproduct;
10092 }
10093
10094 }
10095
10096 if(paneltype && checkexpand) {
10097 /* If any of the Established low-res modes is supported, the
10098 * panel can scale automatically. For 800x600 panels, we only
10099 * check the even lower ones.
10100 */
10101 if(paneltype == Panel_800x600) {
10102 if(buffer[0x23] & 0xfc) cr37 |= 0x10;
10103 } else {
10104 if(buffer[0x23]) cr37 |= 0x10;
10105 }
10106 }
10107
10108 break;
10109
10110 case 3: /* Analyze EDID V2 */
10111 case 4:
10112 index = 0;
10113
10114 if(!(checkedid2(buffer))) {
10115 xf86DrvMsg(pSiS->pScrn->scrnIndex, X_PROBED,
10116 "LCD sense: EDID corrupt\n");
10117 return 0;
10118 }
10119
10120 if((buffer[0x41] & 0x0f) == 0x03) {
10121 index = 0x42 + 3;
10122 xf86DrvMsg(pSiS->pScrn->scrnIndex, X_PROBED,
10123 "LCD sense: Display supports TMDS input on primary interface\n");
10124 } else if((buffer[0x41] & 0xf0) == 0x30) {
10125 index = 0x46 + 3;
10126 xf86DrvMsg(pSiS->pScrn->scrnIndex, X_PROBED,
10127 "LCD sense: Display supports TMDS input on secondary interface\n");
10128 } else {
10129 xf86DrvMsg(pSiS->pScrn->scrnIndex, X_PROBED,
10130 "LCD sense: Display does not support TMDS video interface (0x%02x)\n",
10131 buffer[0x41]);
10132 return 0;
10133 }
10134
10135 SiS_Pr->CP_Vendor = panelvendor = buffer[2] | (buffer[1] << 8);
10136 SiS_Pr->CP_Product = panelproduct = buffer[3] | (buffer[4] << 8);
10137
10138 paneltype = Panel_Custom;
10139 SiS_Pr->CP_MaxX = SiS_Pr->CP_PreferredX = xres = buffer[0x76] | (buffer[0x77] << 8);
10140 SiS_Pr->CP_MaxY = SiS_Pr->CP_PreferredY = yres = buffer[0x78] | (buffer[0x79] << 8);
10141
10142 switch(xres) {
10143#if 0
10144 case 800:
10145 if(yres == 600) {
10146 paneltype = Panel_800x600;
10147 checkexpand = TRUE;
10148 }
10149 break;
10150#endif
10151 case 1024:
10152 if(yres == 768) {
10153 paneltype = Panel_1024x768;
10154 checkexpand = TRUE;
10155 }
10156 break;
10157 case 1280:
10158 if(yres == 960) {
10159 if(pSiS->VGAEngine == SIS_315_VGA) {
10160 paneltype = Panel310_1280x960;
10161 } else {
10162 paneltype = Panel300_1280x960;
10163 }
10164 } else if(yres == 1024) {
10165 paneltype = Panel_1280x1024;
10166 checkexpand = TRUE;
10167 }
10168 /* 1280x768 treated as custom here */
10169 break;
10170 case 1400:
10171 if(pSiS->VGAEngine == SIS_315_VGA) {
10172 if(yres == 1050) {
10173 paneltype = Panel310_1400x1050;
10174 checkexpand = TRUE;
10175 }
10176 }
10177 break;
10178 case 1600:
10179 if(pSiS->VGAEngine == SIS_315_VGA) {
10180 if(pSiS->VBFlags & VB_301C) {
10181 if(yres == 1200) {
10182 paneltype = Panel310_1600x1200;
10183 checkexpand = TRUE;
10184 }
10185 }
10186 }
10187 break;
10188 }
10189
10190 /* Determine if RGB18 or RGB24 */
10191 if(index) {
10192 if((buffer[index] == 0x20) || (buffer[index] == 0x34)) {
10193 cr37 |= 0x01;
10194 }
10195 }
10196
10197 if(checkexpand) {
10198 /* TODO - for now, we let the panel scale */
10199 cr37 |= 0x10;
10200 }
10201
10202 /* Now seek 4-Byte Timing codes and extract sync pol info */
10203 index = 0x80;
10204 if(buffer[0x7e] & 0x20) { /* skip Luminance Table (if provided) */
10205 lumsize = buffer[0x80] & 0x1f;
10206 if(buffer[0x80] & 0x80) lumsize *= 3;
10207 lumsize++; /* luminance header byte */
10208 index += lumsize;
10209 }
10210#if 0 /* "pixel rate" = pixel clock? */
10211 if(buffer[0x7e] & 0x1c) {
10212 for(i=0; i<((buffer[0x7e] & 0x1c) >> 2); i++) {
10213 if(buffer[index + (i*8) + 6] && (buffer[index + (i*8) + 7] & 0x0f)) {
10214 int clk = (buffer[index + (i*8) + 6] | ((buffer[index + (i*8) + 7] & 0x0f) << 4)) * 1000;
10215 if(clk > SiS_Pr->CP_MaxClock) SiS_Pr->CP_MaxClock = clk;
10216 }
10217 }
10218 }
10219#endif
10220 index += (((buffer[0x7e] & 0x1c) >> 2) * 8); /* skip Frequency Ranges */
10221 if(buffer[0x7e] & 0x03) {
10222 for(i=0; i<(buffer[0x7e] & 0x03); i++) {
10223 if((buffer[index + (i*27) + 9]) || (buffer[index + (i*27) + 10])) {
10224 int clk = ((buffer[index + (i*27) + 9]) | ((buffer[index + (i*27) + 9]) << 8)) * 10;
10225 if(clk > SiS_Pr->CP_MaxClock) SiS_Pr->CP_MaxClock = clk;
10226 }
10227 }
10228 }
10229 index += ((buffer[0x7e] & 0x03) * 27); /* skip Detailed Range Limits */
10230 numcodes = (buffer[0x7f] & 0xf8) >> 3;
10231 if(numcodes) {
10232 myindex = index;
10233 seekcode = (xres - 256) / 16;
10234 for(i=0; i<numcodes; i++) {
10235 if(buffer[myindex] == seekcode) break;
10236 myindex += 4;
10237 }
10238 if(buffer[myindex] == seekcode) {
10239 cr37 |= ((((buffer[myindex + 1] & 0x0c) ^ 0x0c) << 4) | 0x20);
10240 havesync = TRUE;
10241 } else {
10242 xf86DrvMsg(pSiS->pScrn->scrnIndex, X_WARNING,
10243 "LCD sense: Unable to retrieve Sync polarity information\n");
10244 }
10245 } else {
10246 xf86DrvMsg(pSiS->pScrn->scrnIndex, X_WARNING,
10247 "LCD sense: Unable to retrieve Sync polarity information\n");
10248 }
10249
10250 /* Check against our database; Eg. Sanyo projector reports
10251 * 1024x768 in non-HDPC mode, although it supports 1280x720.
10252 * Treat such wrongly reporting panels as custom.
10253 */
10254 if(paneltype != Panel_Custom) {
10255 int maxx, maxy, prefx, prefy;
10256 if((SiS_FindPanelFromDB(pSiS, panelvendor, panelproduct, &maxx, &maxy, &prefx, &prefy))) {
10257 paneltype = Panel_Custom;
10258 SiS_Pr->CP_MaxX = maxx;
10259 SiS_Pr->CP_MaxY = maxy;
10260 cr37 |= 0x10;
10261 /* Leave preferred unchanged (MUST be a valid mode!) */
10262 }
10263 }
10264
10265 /* Now seek the detailed timing descriptions for custom panels */
10266 if(paneltype == Panel_Custom) {
10267
10268 SiS_Pr->CP_Supports64048075 = TRUE;
10269
10270 index += (numcodes * 4);
10271 numcodes = buffer[0x7f] & 0x07;
10272 for(i=0; i<numcodes; i++, index += 18) {
10273 xres = buffer[index+2] | ((buffer[index+4] & 0xf0) << 4);
10274 yres = buffer[index+5] | ((buffer[index+7] & 0xf0) << 4);
10275
10276 SiS_Pr->CP_HDisplay[i] = xres;
10277 SiS_Pr->CP_HSyncStart[i] = xres + (buffer[index+8] | ((buffer[index+11] & 0xc0) << 2));
10278 SiS_Pr->CP_HSyncEnd[i] = SiS_Pr->CP_HSyncStart[i] + (buffer[index+9] | ((buffer[index+11] & 0x30) << 4));
10279 SiS_Pr->CP_HTotal[i] = xres + (buffer[index+3] | ((buffer[index+4] & 0x0f) << 8));
10280 SiS_Pr->CP_HBlankStart[i] = xres + 1;
10281 SiS_Pr->CP_HBlankEnd[i] = SiS_Pr->CP_HTotal[i];
10282
10283 SiS_Pr->CP_VDisplay[i] = yres;
10284 SiS_Pr->CP_VSyncStart[i] = yres + (((buffer[index+10] & 0xf0) >> 4) | ((buffer[index+11] & 0x0c) << 2));
10285 SiS_Pr->CP_VSyncEnd[i] = SiS_Pr->CP_VSyncStart[i] + ((buffer[index+10] & 0x0f) | ((buffer[index+11] & 0x03) << 4));
10286 SiS_Pr->CP_VTotal[i] = yres + (buffer[index+6] | ((buffer[index+7] & 0x0f) << 8));
10287 SiS_Pr->CP_VBlankStart[i] = yres + 1;
10288 SiS_Pr->CP_VBlankEnd[i] = SiS_Pr->CP_VTotal[i];
10289
10290 SiS_Pr->CP_Clock[i] = (buffer[index] | (buffer[index+1] << 8)) * 10;
10291
10292 SiS_Pr->CP_DataValid[i] = TRUE;
10293
10294 if((SiS_Pr->CP_HDisplay[i] & 7) ||
10295 (SiS_Pr->CP_HDisplay[i] > SiS_Pr->CP_HSyncStart[i]) ||
10296 (SiS_Pr->CP_HDisplay[i] >= SiS_Pr->CP_HSyncEnd[i]) ||
10297 (SiS_Pr->CP_HDisplay[i] >= SiS_Pr->CP_HTotal[i]) ||
10298 (SiS_Pr->CP_HSyncStart[i] >= SiS_Pr->CP_HSyncEnd[i]) ||
10299 (SiS_Pr->CP_HSyncStart[i] > SiS_Pr->CP_HTotal[i]) ||
10300 (SiS_Pr->CP_HSyncEnd[i] > SiS_Pr->CP_HTotal[i]) ||
10301 (SiS_Pr->CP_VDisplay[i] > SiS_Pr->CP_VSyncStart[i]) ||
10302 (SiS_Pr->CP_VDisplay[i] >= SiS_Pr->CP_VSyncEnd[i]) ||
10303 (SiS_Pr->CP_VDisplay[i] >= SiS_Pr->CP_VTotal[i]) ||
10304 (SiS_Pr->CP_VSyncStart[i] > SiS_Pr->CP_VSyncEnd[i]) ||
10305 (SiS_Pr->CP_VSyncStart[i] > SiS_Pr->CP_VTotal[i]) ||
10306 (SiS_Pr->CP_VSyncEnd[i] > SiS_Pr->CP_VTotal[i]) ||
10307 (((pSiS->VBFlags & VB_301C) && (SiS_Pr->CP_Clock[i] > 162500)) ||
10308 ((!(pSiS->VBFlags & VB_301C)) &&
10309 ((SiS_Pr->CP_Clock[i] > 108200) || (SiS_Pr->CP_VDisplay[i] > 1024)))) ||
10310 (buffer[index + 17] & 0x80)) {
10311
10312 SiS_Pr->CP_DataValid[i] = FALSE;
10313
10314 } else {
10315
10316 SiS_Pr->CP_HaveCustomData = TRUE;
10317
10318 if(SiS_Pr->CP_Clock[i] > SiS_Pr->CP_MaxClock) SiS_Pr->CP_MaxClock = SiS_Pr->CP_Clock[i];
10319
10320 if((SiS_Pr->CP_PreferredX == xres) && (SiS_Pr->CP_PreferredY == yres)) {
10321 SiS_Pr->CP_PreferredIndex = i;
10322 SiS_MakeClockRegs(pSiS->pScrn, SiS_Pr->CP_Clock[i], &SiS_Pr->CP_PrefSR2B, &SiS_Pr->CP_PrefSR2C);
10323 SiS_Pr->CP_PrefClock = (SiS_Pr->CP_Clock[i] / 1000) + 1;
10324 if(!havesync) {
10325 cr37 |= ((((buffer[index + 17] & 0x06) ^ 0x06) << 5) | 0x20);
10326 havesync = TRUE;
10327 }
10328 }
10329
10330 SiS_Pr->CP_HSync_P[i] = (buffer[index + 17] & 0x02) ? TRUE : FALSE;
10331 SiS_Pr->CP_VSync_P[i] = (buffer[index + 17] & 0x04) ? TRUE : FALSE;
10332 SiS_Pr->CP_SyncValid[i] = TRUE;
10333
10334 }
10335 }
10336
10337 cr37 |= 0x10;
10338
10339 }
10340
10341 break;
10342
10343 }
10344
10345 /* 1280x960 panels are always RGB24, unable to scale and use
10346 * high active sync polarity
10347 */
10348 if(pSiS->VGAEngine == SIS_315_VGA) {
10349 if(paneltype == Panel310_1280x960) cr37 &= 0x0e;
10350 } else {
10351 if(paneltype == Panel300_1280x960) cr37 &= 0x0e;
10352 }
10353
10354 for(i = 0; i < 7; i++) {
10355 if(SiS_Pr->CP_DataValid[i]) {
10356 xf86DrvMsg(pSiS->pScrn->scrnIndex, X_PROBED,
10357 "Non-standard LCD/DVI-D timing data no. %d:\n", i);
10358 xf86DrvMsg(pSiS->pScrn->scrnIndex, X_PROBED,
10359 " HDisplay %d HSync %d HSyncEnd %d HTotal %d\n",
10360 SiS_Pr->CP_HDisplay[i], SiS_Pr->CP_HSyncStart[i],
10361 SiS_Pr->CP_HSyncEnd[i], SiS_Pr->CP_HTotal[i]);
10362 xf86DrvMsg(pSiS->pScrn->scrnIndex, X_PROBED,
10363 " VDisplay %d VSync %d VSyncEnd %d VTotal %d\n",
10364 SiS_Pr->CP_VDisplay[i], SiS_Pr->CP_VSyncStart[i],
10365 SiS_Pr->CP_VSyncEnd[i], SiS_Pr->CP_VTotal[i]);
10366 xf86DrvMsg(pSiS->pScrn->scrnIndex, X_PROBED,
10367 " Pixel clock: %3.3fMhz\n", (float)SiS_Pr->CP_Clock[i] / 1000);
10368 xf86DrvMsg(pSiS->pScrn->scrnIndex, X_INFO,
10369 " To use this, add \"%dx%d\" to the list of Modes in the Screen section\n",
10370 SiS_Pr->CP_HDisplay[i],
10371 SiS_Pr->CP_VDisplay[i]);
10372 }
10373 }
10374
10375 if(paneltype) {
10376 if(!SiS_Pr->CP_PreferredX) SiS_Pr->CP_PreferredX = SiS_Pr->CP_MaxX;
10377 if(!SiS_Pr->CP_PreferredY) SiS_Pr->CP_PreferredY = SiS_Pr->CP_MaxY;
10378 SiS_SetRegOR(SiS_Pr->SiS_P3d4,0x32,0x08);
10379 SiS_SetReg(SiS_Pr->SiS_P3d4,0x36,paneltype);
10380 cr37 &= 0xf1;
10381 SiS_SetRegANDOR(SiS_Pr->SiS_P3d4,0x37,0x0c,cr37);
10382 SiS_Pr->PanelSelfDetected = TRUE;
10383#ifdef TWDEBUG
10384 xf86DrvMsgVerb(pSiS->pScrn->scrnIndex, X_PROBED, 3,
10385 "LCD sense: [DDC LCD results: 0x%02x, 0x%02x]\n", paneltype, cr37);
10386#endif
10387 } else {
10388 SiS_SetRegAND(SiS_Pr->SiS_P3d4,0x32,~0x08);
10389 SiS_SetReg(SiS_Pr->SiS_P3d4,0x36,0x00);
10390 }
10391 return 0;
10392}
10393
10394USHORT
10395SiS_SenseVGA2DDC(SiS_Private *SiS_Pr, SISPtr pSiS)
10396{
10397 USHORT DDCdatatype,flag;
10398 BOOLEAN foundcrt = FALSE;
10399 int retry;
10400 unsigned char buffer[256];
10401
10402 if(!(pSiS->VBFlags & (VB_301|VB_301B|VB_301C|VB_302B))) return 0;
10403
10404 if(SiS_InitDDCRegs(SiS_Pr, pSiS->VBFlags, pSiS->VGAEngine, 2, 0, FALSE) == 0xFFFF) return 0;
10405
10406 SiS_Pr->SiS_DDC_SecAddr = 0x00;
10407
10408 /* Probe supported DA's */
10409 flag = SiS_ProbeDDC(SiS_Pr);
10410 if(flag & 0x10) {
10411 SiS_Pr->SiS_DDC_DeviceAddr = 0xa6; /* EDID V2 (FP) */
10412 DDCdatatype = 4;
10413 } else if(flag & 0x08) {
10414 SiS_Pr->SiS_DDC_DeviceAddr = 0xa2; /* EDID V2 (P&D-D Monitor) */
10415 DDCdatatype = 3;
10416 } else if(flag & 0x02) {
10417 SiS_Pr->SiS_DDC_DeviceAddr = 0xa0; /* EDID V1 */
10418 DDCdatatype = 1;
10419 } else {
10420 xf86DrvMsg(pSiS->pScrn->scrnIndex, X_PROBED,
10421 "VGA2 sense: Do DDC answer\n");
10422 return 0; /* no DDC support (or no device attached) */
10423 }
10424
10425 /* Read the entire EDID */
10426 retry = 2;
10427 do {
10428 if(SiS_ReadDDC(SiS_Pr, DDCdatatype, buffer)) {
10429 xf86DrvMsg(pSiS->pScrn->scrnIndex, X_PROBED,
10430 "VGA2 sense: DDC read failed (attempt %d), %s\n",
10431 (3-retry), (retry == 1) ? "giving up" : "retrying");
10432 retry--;
10433 if(retry == 0) return 0xFFFF;
10434 } else break;
10435 } while(1);
10436
10437 /* Analyze EDID. We don't have many chances to
10438 * distinguish a flat panel from a CRT...
10439 */
10440 switch(DDCdatatype) {
10441 case 1:
10442 if(!(checkedid1(buffer))) {
10443 xf86DrvMsg(pSiS->pScrn->scrnIndex, X_ERROR,
10444 "VGA2 sense: EDID corrupt\n");
10445 return 0;
10446 }
10447 if(buffer[0x14] & 0x80) { /* Display uses digital input */
10448 xf86DrvMsg(pSiS->pScrn->scrnIndex, X_ERROR,
10449 "VGA2 sense: Attached display expects digital input\n");
10450 return 0;
10451 }
10452 SiS_Pr->CP_Vendor = buffer[9] | (buffer[8] << 8);
10453 SiS_Pr->CP_Product = buffer[10] | (buffer[11] << 8);
10454 foundcrt = TRUE;
10455 break;
10456 case 3:
10457 case 4:
10458 if(!(checkedid2(buffer))) {
10459 xf86DrvMsg(pSiS->pScrn->scrnIndex, X_ERROR,
10460 "VGA2 sense: EDID corrupt\n");
10461 return 0;
10462 }
10463 if( ((buffer[0x41] & 0x0f) != 0x01) && /* Display does not support analog input */
10464 ((buffer[0x41] & 0x0f) != 0x02) &&
10465 ((buffer[0x41] & 0xf0) != 0x10) &&
10466 ((buffer[0x41] & 0xf0) != 0x20) ) {
10467 xf86DrvMsg(pSiS->pScrn->scrnIndex, X_ERROR,
10468 "VGA2 sense: Attached display does not support analog input (0x%02x)\n",
10469 buffer[0x41]);
10470 return 0;
10471 }
10472 SiS_Pr->CP_Vendor = buffer[2] | (buffer[1] << 8);
10473 SiS_Pr->CP_Product = buffer[3] | (buffer[4] << 8);
10474 foundcrt = TRUE;
10475 break;
10476 }
10477
10478 if(foundcrt) {
10479 SiS_SetRegOR(SiS_Pr->SiS_P3d4,0x32,0x10);
10480 }
10481 return(0);
10482}
10483
10484#endif
10485
10486void
10487SiS_SetCH70xxANDOR(SiS_Private *SiS_Pr, USHORT tempax,USHORT tempbh)
10488{
10489 USHORT tempbl;
10490
10491 tempbl = SiS_GetCH70xx(SiS_Pr,(tempax & 0x00FF));
10492 tempbl = (((tempbl & tempbh) << 8) | tempax);
10493 SiS_SetCH70xx(SiS_Pr,tempbl);
10494}
10495
10496/* Generic I2C functions for Chrontel & DDC --------- */
10497
10498static void
10499SiS_SetSwitchDDC2(SiS_Private *SiS_Pr)
10500{
10501 SiS_SetSCLKHigh(SiS_Pr);
10502 SiS_WaitRetrace1(SiS_Pr);
10503
10504 SiS_SetSCLKLow(SiS_Pr);
10505 SiS_WaitRetrace1(SiS_Pr);
10506}
10507
10508USHORT
10509SiS_ReadDDC1Bit(SiS_Private *SiS_Pr)
10510{
10511 SiS_WaitRetrace1(SiS_Pr);
10512 return((SiS_GetReg(SiS_Pr->SiS_P3c4,0x11) & 0x02) >> 1);
10513}
10514
10515/* Set I2C start condition */
10516/* This is done by a SD high-to-low transition while SC is high */
10517static USHORT
10518SiS_SetStart(SiS_Private *SiS_Pr)
10519{
10520 if(SiS_SetSCLKLow(SiS_Pr)) return 0xFFFF; /* (SC->low) */
10521 SiS_SetRegANDOR(SiS_Pr->SiS_DDC_Port,
10522 SiS_Pr->SiS_DDC_Index,
10523 SiS_Pr->SiS_DDC_NData,
10524 SiS_Pr->SiS_DDC_Data); /* SD->high */
10525 if(SiS_SetSCLKHigh(SiS_Pr)) return 0xFFFF; /* SC->high */
10526 SiS_SetRegANDOR(SiS_Pr->SiS_DDC_Port,
10527 SiS_Pr->SiS_DDC_Index,
10528 SiS_Pr->SiS_DDC_NData,
10529 0x00); /* SD->low = start condition */
10530 if(SiS_SetSCLKHigh(SiS_Pr)) return 0xFFFF; /* (SC->low) */
10531 return 0;
10532}
10533
10534/* Set I2C stop condition */
10535/* This is done by a SD low-to-high transition while SC is high */
10536static USHORT
10537SiS_SetStop(SiS_Private *SiS_Pr)
10538{
10539 if(SiS_SetSCLKLow(SiS_Pr)) return 0xFFFF; /* (SC->low) */
10540 SiS_SetRegANDOR(SiS_Pr->SiS_DDC_Port,
10541 SiS_Pr->SiS_DDC_Index,
10542 SiS_Pr->SiS_DDC_NData,
10543 0x00); /* SD->low */
10544 if(SiS_SetSCLKHigh(SiS_Pr)) return 0xFFFF; /* SC->high */
10545 SiS_SetRegANDOR(SiS_Pr->SiS_DDC_Port,
10546 SiS_Pr->SiS_DDC_Index,
10547 SiS_Pr->SiS_DDC_NData,
10548 SiS_Pr->SiS_DDC_Data); /* SD->high = stop condition */
10549 if(SiS_SetSCLKHigh(SiS_Pr)) return 0xFFFF; /* (SC->high) */
10550 return 0;
10551}
10552
10553/* Write 8 bits of data */
10554static USHORT
10555SiS_WriteDDC2Data(SiS_Private *SiS_Pr, USHORT tempax)
10556{
10557 USHORT i,flag,temp;
10558
10559 flag = 0x80;
10560 for(i=0; i<8; i++) {
10561 SiS_SetSCLKLow(SiS_Pr); /* SC->low */
10562 if(tempax & flag) {
10563 SiS_SetRegANDOR(SiS_Pr->SiS_DDC_Port,
10564 SiS_Pr->SiS_DDC_Index,
10565 SiS_Pr->SiS_DDC_NData,
10566 SiS_Pr->SiS_DDC_Data); /* Write bit (1) to SD */
10567 } else {
10568 SiS_SetRegANDOR(SiS_Pr->SiS_DDC_Port,
10569 SiS_Pr->SiS_DDC_Index,
10570 SiS_Pr->SiS_DDC_NData,
10571 0x00); /* Write bit (0) to SD */
10572 }
10573 SiS_SetSCLKHigh(SiS_Pr); /* SC->high */
10574 flag >>= 1;
10575 }
10576 temp = SiS_CheckACK(SiS_Pr); /* Check acknowledge */
10577 return(temp);
10578}
10579
10580static USHORT
10581SiS_ReadDDC2Data(SiS_Private *SiS_Pr, USHORT tempax)
10582{
10583 USHORT i,temp,getdata;
10584
10585 getdata=0;
10586 for(i=0; i<8; i++) {
10587 getdata <<= 1;
10588 SiS_SetSCLKLow(SiS_Pr);
10589 SiS_SetRegANDOR(SiS_Pr->SiS_DDC_Port,
10590 SiS_Pr->SiS_DDC_Index,
10591 SiS_Pr->SiS_DDC_NData,
10592 SiS_Pr->SiS_DDC_Data);
10593 SiS_SetSCLKHigh(SiS_Pr);
10594 temp = SiS_GetReg(SiS_Pr->SiS_DDC_Port,SiS_Pr->SiS_DDC_Index);
10595 if(temp & SiS_Pr->SiS_DDC_Data) getdata |= 0x01;
10596 }
10597 return(getdata);
10598}
10599
10600static USHORT
10601SiS_SetSCLKLow(SiS_Private *SiS_Pr)
10602{
10603 SiS_SetRegANDOR(SiS_Pr->SiS_DDC_Port,
10604 SiS_Pr->SiS_DDC_Index,
10605 SiS_Pr->SiS_DDC_NClk,
10606 0x00); /* SetSCLKLow() */
10607 SiS_DDC2Delay(SiS_Pr,SiS_I2CDELAYSHORT);
10608 return 0;
10609}
10610
10611static USHORT
10612SiS_SetSCLKHigh(SiS_Private *SiS_Pr)
10613{
10614 USHORT temp, watchdog=1000;
10615
10616 SiS_SetRegANDOR(SiS_Pr->SiS_DDC_Port,
10617 SiS_Pr->SiS_DDC_Index,
10618 SiS_Pr->SiS_DDC_NClk,
10619 SiS_Pr->SiS_DDC_Clk); /* SetSCLKHigh() */
10620 do {
10621 temp = SiS_GetReg(SiS_Pr->SiS_DDC_Port,SiS_Pr->SiS_DDC_Index);
10622 } while((!(temp & SiS_Pr->SiS_DDC_Clk)) && --watchdog);
10623 if (!watchdog) {
10624#ifdef TWDEBUG
10625 xf86DrvMsg(0, X_INFO, "SetClkHigh failed\n");
10626#endif
10627 return 0xFFFF;
10628 }
10629 SiS_DDC2Delay(SiS_Pr,SiS_I2CDELAYSHORT);
10630 return 0;
10631}
10632
10633/* Check I2C acknowledge */
10634/* Returns 0 if ack ok, non-0 if ack not ok */
10635static USHORT
10636SiS_CheckACK(SiS_Private *SiS_Pr)
10637{
10638 USHORT tempah;
10639
10640 SiS_SetSCLKLow(SiS_Pr); /* (SC->low) */
10641 SiS_SetRegANDOR(SiS_Pr->SiS_DDC_Port,
10642 SiS_Pr->SiS_DDC_Index,
10643 SiS_Pr->SiS_DDC_NData,
10644 SiS_Pr->SiS_DDC_Data); /* (SD->high) */
10645 SiS_SetSCLKHigh(SiS_Pr); /* SC->high = clock impulse for ack */
10646 tempah = SiS_GetReg(SiS_Pr->SiS_DDC_Port,SiS_Pr->SiS_DDC_Index); /* Read SD */
10647 SiS_SetSCLKLow(SiS_Pr); /* SC->low = end of clock impulse */
10648 if(tempah & SiS_Pr->SiS_DDC_Data) return(1); /* Ack OK if bit = 0 */
10649 else return(0);
10650}
10651
10652/* End of I2C functions ----------------------- */
10653
10654
10655/* =============== SiS 315/330 O.E.M. ================= */
10656
10657#ifdef SIS315H
10658
10659static USHORT
10660GetRAMDACromptr(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
10661{
10662 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
10663 USHORT romptr;
10664
10665 if(HwInfo->jChipType < SIS_330) {
10666 romptr = SISGETROMW(0x128);
10667 if(SiS_Pr->SiS_VBType & VB_SIS301B302B)
10668 romptr = SISGETROMW(0x12a);
10669 } else {
10670 romptr = SISGETROMW(0x1a8);
10671 if(SiS_Pr->SiS_VBType & VB_SIS301B302B)
10672 romptr = SISGETROMW(0x1aa);
10673 }
10674 return(romptr);
10675}
10676
10677static USHORT
10678GetLCDromptr(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
10679{
10680 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
10681 USHORT romptr;
10682
10683 if(HwInfo->jChipType < SIS_330) {
10684 romptr = SISGETROMW(0x120);
10685 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV)
10686 romptr = SISGETROMW(0x122);
10687 } else {
10688 romptr = SISGETROMW(0x1a0);
10689 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV)
10690 romptr = SISGETROMW(0x1a2);
10691 }
10692 return(romptr);
10693}
10694
10695static USHORT
10696GetTVromptr(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
10697{
10698 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
10699 USHORT romptr;
10700
10701 if(HwInfo->jChipType < SIS_330) {
10702 romptr = SISGETROMW(0x114);
10703 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV)
10704 romptr = SISGETROMW(0x11a);
10705 } else {
10706 romptr = SISGETROMW(0x194);
10707 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV)
10708 romptr = SISGETROMW(0x19a);
10709 }
10710 return(romptr);
10711}
10712
10713static USHORT
10714GetLCDPtrIndexBIOS(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
10715{
10716 USHORT index;
10717
10718 if((IS_SIS650) && (SiS_Pr->SiS_VBType & VB_SIS301LV302LV)) {
10719 if(!(SiS_IsNotM650orLater(SiS_Pr, HwInfo))) {
10720 if((index = SiS_GetReg(SiS_Pr->SiS_P3d4,0x36) & 0xf0)) {
10721 index >>= 4;
10722 index *= 3;
10723 if(SiS_Pr->SiS_LCDInfo & DontExpandLCD) index += 2;
10724 else if(!(SiS_Pr->SiS_SetFlag & LCDVESATiming)) index++;
10725 return index;
10726 }
10727 }
10728 }
10729
10730 index = SiS_GetBIOSLCDResInfo(SiS_Pr) & 0x0F;
10731 if(SiS_Pr->SiS_LCDResInfo == Panel_1400x1050) index -= 5;
10732 else if(SiS_Pr->SiS_LCDResInfo == Panel_1600x1200) index -= 6;
10733 index--;
10734 index *= 3;
10735 if(SiS_Pr->SiS_LCDInfo & DontExpandLCD) index += 2;
10736 else if(!(SiS_Pr->SiS_SetFlag & LCDVESATiming)) index++;
10737 return index;
10738}
10739
10740static USHORT
10741GetLCDPtrIndex(SiS_Private *SiS_Pr)
10742{
10743 USHORT index;
10744
10745 index = ((SiS_GetBIOSLCDResInfo(SiS_Pr) & 0x0F) - 1) * 3;
10746 if(SiS_Pr->SiS_LCDInfo & DontExpandLCD) index += 2;
10747 else if(!(SiS_Pr->SiS_SetFlag & LCDVESATiming)) index++;
10748 return index;
10749}
10750
10751static USHORT
10752GetTVPtrIndex(SiS_Private *SiS_Pr)
10753{
10754 USHORT index;
10755
10756 index = 0;
10757 if(SiS_Pr->SiS_TVMode & TVSetPAL) index = 1;
10758 if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) index = 2;
10759
10760 if(SiS_Pr->SiS_VBInfo & SetCRT2ToYPbPr525750) index = 0;
10761
10762 index <<= 1;
10763
10764 if((SiS_Pr->SiS_VBInfo & SetInSlaveMode) &&
10765 (SiS_Pr->SiS_TVMode & TVSetTVSimuMode)) {
10766 index++;
10767 }
10768
10769 return index;
10770}
10771
10772static ULONG
10773GetOEMTVPtr661_2_GEN(SiS_Private *SiS_Pr, int addme)
10774{
10775 USHORT index = 0, temp = 0;
10776
10777 if(SiS_Pr->SiS_TVMode & TVSetPAL) index = 1;
10778 if(SiS_Pr->SiS_TVMode & TVSetPALM) index = 2;
10779 if(SiS_Pr->SiS_TVMode & TVSetPALN) index = 3;
10780 if(SiS_Pr->SiS_TVMode & TVSetNTSCJ) index = 6;
10781 if(SiS_Pr->SiS_TVMode & TVSetNTSC1024) {
10782 index = 4;
10783 if(SiS_Pr->SiS_TVMode & TVSetPALM) index++;
10784 if(SiS_Pr->SiS_TVMode & TVSetNTSCJ) index = 7;
10785 }
10786
10787 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) {
10788 if((!(SiS_Pr->SiS_VBInfo & SetInSlaveMode)) ||
10789 (SiS_Pr->SiS_TVMode & TVSetTVSimuMode)) {
10790 index += addme;
10791 temp++;
10792 }
10793 temp += 0x0100;
10794 }
10795 return(ULONG)(index | (temp << 16));
10796}
10797
10798static ULONG
10799GetOEMTVPtr661_2_OLD(SiS_Private *SiS_Pr)
10800{
10801 return(GetOEMTVPtr661_2_GEN(SiS_Pr, 8));
10802}
10803
10804#if 0
10805static ULONG
10806GetOEMTVPtr661_2_NEW(SiS_Private *SiS_Pr)
10807{
10808 return(GetOEMTVPtr661_2_GEN(SiS_Pr, 6));
10809}
10810#endif
10811
10812static int
10813GetOEMTVPtr661(SiS_Private *SiS_Pr)
10814{
10815 int index = 0;
10816
10817 if(SiS_Pr->SiS_TVMode & TVSetPAL) index = 2;
10818 if(SiS_Pr->SiS_ROMNew) {
10819 if(SiS_Pr->SiS_TVMode & TVSetYPbPr525i) index = 4;
10820 if(SiS_Pr->SiS_TVMode & TVSetYPbPr525p) index = 6;
10821 if(SiS_Pr->SiS_TVMode & TVSetYPbPr750p) index = 8;
10822 if(SiS_Pr->SiS_TVMode & TVSetHiVision) index = 10;
10823 } else {
10824 if(SiS_Pr->SiS_TVMode & TVSetHiVision) index = 4;
10825 if(SiS_Pr->SiS_TVMode & TVSetYPbPr525i) index = 6;
10826 if(SiS_Pr->SiS_TVMode & TVSetYPbPr525p) index = 8;
10827 if(SiS_Pr->SiS_TVMode & TVSetYPbPr750p) index = 10;
10828 }
10829
10830 if(SiS_Pr->SiS_TVMode & TVSetTVSimuMode) index++;
10831
10832 return index;
10833}
10834
10835static void
10836SetDelayComp(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo, USHORT ModeNo)
10837{
10838 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
10839 USHORT delay=0,index,myindex,temp,romptr=0;
10840 BOOLEAN dochiptest = TRUE;
10841
10842 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA) {
10843 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x20,0xbf);
10844 } else {
10845 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x35,0x7f);
10846 }
10847
10848 /* Find delay (from ROM, internal tables, PCI subsystem) */
10849
10850 if(SiS_Pr->SiS_VBInfo & SetCRT2ToRAMDAC) { /* ------------ VGA */
10851
10852 if((SiS_Pr->SiS_UseROM) && (!(SiS_Pr->SiS_ROMNew))) {
10853 romptr = GetRAMDACromptr(SiS_Pr, HwInfo);
10854 }
10855 if(romptr) delay = ROMAddr[romptr];
10856 else {
10857 delay = 0x04;
10858 if(SiS_Pr->SiS_VBType & VB_SIS301B302B) {
10859 if(IS_SIS650) {
10860 delay = 0x0a;
10861 } else if(IS_SIS740) {
10862 delay = 0x00;
10863 } else if(HwInfo->jChipType < SIS_330) {
10864 delay = 0x0c;
10865 } else {
10866 delay = 0x0c;
10867 }
10868 } else if(SiS_Pr->SiS_IF_DEF_LVDS == 1) {
10869 delay = 0x00;
10870 }
10871 }
10872
10873 } else if(SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD|SetCRT2ToLCDA)) { /* ---------- LCD/LCDA */
10874
10875 BOOLEAN gotitfrompci = FALSE;
10876
10877 /* Could we detect a PDC for LCD or did we get a user-defined? If yes, use it */
10878
10879 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) {
10880 if(SiS_Pr->PDC != -1) {
10881 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x2d,0xf0,((SiS_Pr->PDC >> 1) & 0x0f));
10882 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x35,0x7f,((SiS_Pr->PDC & 0x01) << 7));
10883 return;
10884 }
10885 } else {
10886 if(SiS_Pr->PDCA != -1) {
10887 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x2d,0x0f,((SiS_Pr->PDCA << 3) & 0xf0));
10888 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x20,0xbf,((SiS_Pr->PDCA & 0x01) << 6));
10889 return;
10890 }
10891 }
10892
10893 /* Custom Panel? */
10894
10895 if(SiS_Pr->SiS_LCDResInfo == Panel_Custom) {
10896 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA) {
10897 delay = 0x00;
10898 if((SiS_Pr->PanelXRes <= 1280) && (SiS_Pr->PanelYRes <= 1024)) {
10899 delay = 0x20;
10900 }
10901 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x2d,0x0f,delay);
10902 } else {
10903 delay = 0x0c;
10904 if(SiS_Pr->SiS_VBType & VB_SIS301C) delay = 0x03;
10905 else if(SiS_Pr->SiS_VBType & VB_SIS301LV302LV) {
10906 if(IS_SIS740) delay = 0x01;
10907 else delay = 0x03;
10908 }
10909 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x2d,0xf0,delay);
10910 }
10911 return;
10912 }
10913
10914 /* This is a piece of typical SiS crap: They code the OEM LCD
10915 * delay into the code, at no defined place in the BIOS.
10916 * We now have to start doing a PCI subsystem check here.
10917 */
10918
10919 switch(SiS_Pr->SiS_CustomT) {
10920 case CUT_COMPAQ1280:
10921 case CUT_COMPAQ12802:
10922 if(SiS_Pr->SiS_LCDResInfo == Panel_1280x1024) {
10923 gotitfrompci = TRUE;
10924 dochiptest = FALSE;
10925 delay = 0x03;
10926 }
10927 break;
10928 case CUT_CLEVO1400:
10929 case CUT_CLEVO14002:
10930 gotitfrompci = TRUE;
10931 dochiptest = FALSE;
10932 delay = 0x02;
10933 break;
10934 case CUT_CLEVO1024:
10935 case CUT_CLEVO10242:
10936 if(SiS_Pr->SiS_LCDResInfo == Panel_1024x768) {
10937 gotitfrompci = TRUE;
10938 dochiptest = FALSE;
10939 delay = 0x33;
10940 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x2D,delay);
10941 delay &= 0x0f;
10942 }
10943 break;
10944 }
10945
10946 /* Could we find it through the PCI ID? If no, use ROM or table */
10947
10948 if(!gotitfrompci) {
10949
10950 index = GetLCDPtrIndexBIOS(SiS_Pr, HwInfo);
10951 myindex = GetLCDPtrIndex(SiS_Pr);
10952
10953 if(IS_SIS650 && (SiS_Pr->SiS_VBType & VB_SIS301LV302LV)) {
10954
10955 if(SiS_IsNotM650orLater(SiS_Pr, HwInfo)) {
10956
10957 if((SiS_Pr->SiS_UseROM) && (!(SiS_Pr->SiS_ROMNew))) {
10958 /* Always use the second pointer on 650; some BIOSes */
10959 /* still carry old 301 data at the first location */
10960 /* romptr = SISGETROMW(0x120); */
10961 /* if(SiS_Pr->SiS_VBType & VB_SIS302LV) */
10962 romptr = SISGETROMW(0x122);
10963 if(!romptr) return;
10964 delay = ROMAddr[(romptr + index)];
10965 } else {
10966 delay = SiS310_LCDDelayCompensation_650301LV[myindex];
10967 }
10968
10969 } else {
10970
10971 delay = SiS310_LCDDelayCompensation_651301LV[myindex];
10972 if(SiS_Pr->SiS_VBType & (VB_SIS302LV | VB_SIS302ELV))
10973 delay = SiS310_LCDDelayCompensation_651302LV[myindex];
10974
10975 }
10976
10977 } else if(SiS_Pr->SiS_UseROM &&
10978 (!(SiS_Pr->SiS_ROMNew)) &&
10979 (SiS_Pr->SiS_LCDResInfo != Panel_1280x1024) &&
10980 (SiS_Pr->SiS_LCDResInfo != Panel_1280x768) &&
10981 (SiS_Pr->SiS_LCDResInfo != Panel_1280x960)) {
10982
10983 /* Data for 1280x1024 wrong in 301B BIOS */
10984 romptr = GetLCDromptr(SiS_Pr, HwInfo);
10985 if(!romptr) return;
10986 delay = ROMAddr[(romptr + index)];
10987
10988 } else if(SiS_Pr->SiS_IF_DEF_LVDS == 1) {
10989
10990 if(IS_SIS740) delay = 0x03;
10991 else delay = 0x00;
10992
10993 } else {
10994
10995 delay = SiS310_LCDDelayCompensation_301[myindex];
10996 if(SiS_Pr->SiS_VBType & VB_SIS301LV302LV) {
10997 if(IS_SIS740) delay = 0x01;
10998 else if(HwInfo->jChipType <= SIS_315PRO) delay = SiS310_LCDDelayCompensation_3xx301LV[myindex];
10999 else delay = SiS310_LCDDelayCompensation_650301LV[myindex];
11000 } else if(SiS_Pr->SiS_VBType & VB_SIS301C) {
11001 if(IS_SIS740) delay = 0x01; /* ? */
11002 else delay = 0x03;
11003 } else if(SiS_Pr->SiS_VBType & VB_SIS301B302B) {
11004 if(IS_SIS740) delay = 0x01;
11005 else delay = SiS310_LCDDelayCompensation_3xx301B[myindex];
11006 }
11007
11008 }
11009
11010 } /* got it from PCI */
11011
11012 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA) {
11013 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x2D,0x0F,((delay << 4) & 0xf0));
11014 dochiptest = FALSE;
11015 }
11016
11017 } else if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) { /* ------------ TV */
11018
11019 index = GetTVPtrIndex(SiS_Pr);
11020
11021 if(IS_SIS650 && (SiS_Pr->SiS_VBType & VB_SIS301LV302LV)) {
11022
11023 if(SiS_IsNotM650orLater(SiS_Pr,HwInfo)) {
11024
11025 if((SiS_Pr->SiS_UseROM) && (!(SiS_Pr->SiS_ROMNew))) {
11026 /* Always use the second pointer on 650; some BIOSes */
11027 /* still carry old 301 data at the first location */
11028 /* romptr = SISGETROMW(0x114); */
11029 /* if(SiS_Pr->SiS_VBType & VB_SIS302LV) */
11030 romptr = SISGETROMW(0x11a);
11031 if(!romptr) return;
11032 delay = ROMAddr[romptr + index];
11033
11034 } else {
11035
11036 delay = SiS310_TVDelayCompensation_301B[index];
11037
11038 }
11039
11040 } else {
11041
11042 switch(SiS_Pr->SiS_CustomT) {
11043 case CUT_COMPAQ1280:
11044 case CUT_COMPAQ12802:
11045 case CUT_CLEVO1400:
11046 case CUT_CLEVO14002:
11047 delay = 0x02;
11048 dochiptest = FALSE;
11049 break;
11050 case CUT_CLEVO1024:
11051 case CUT_CLEVO10242:
11052 delay = 0x03;
11053 dochiptest = FALSE;
11054 break;
11055 default:
11056 delay = SiS310_TVDelayCompensation_651301LV[index];
11057 if(SiS_Pr->SiS_VBType & VB_SIS302LV) {
11058 delay = SiS310_TVDelayCompensation_651302LV[index];
11059 }
11060 }
11061 }
11062
11063 } else if((SiS_Pr->SiS_UseROM) && (!(SiS_Pr->SiS_ROMNew))) {
11064
11065 romptr = GetTVromptr(SiS_Pr, HwInfo);
11066 if(!romptr) return;
11067 delay = ROMAddr[romptr + index];
11068
11069 } else if(SiS_Pr->SiS_IF_DEF_LVDS == 1) {
11070
11071 delay = SiS310_TVDelayCompensation_LVDS[index];
11072
11073 } else {
11074
11075 delay = SiS310_TVDelayCompensation_301[index];
11076 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) {
11077 if(IS_SIS740) {
11078 delay = SiS310_TVDelayCompensation_740301B[index];
11079 /* LV: use 301 data? BIOS bug? */
11080 } else {
11081 delay = SiS310_TVDelayCompensation_301B[index];
11082 if(SiS_Pr->SiS_VBType & VB_SIS301C) delay = 0x02;
11083 }
11084 }
11085
11086 }
11087
11088 if(SiS_LCDAEnabled(SiS_Pr, HwInfo)) {
11089 delay &= 0x0f;
11090 dochiptest = FALSE;
11091 }
11092
11093 } else return;
11094
11095 /* Write delay */
11096
11097 if(SiS_Pr->SiS_VBType & VB_SISVB) {
11098
11099 if(IS_SIS650 && (SiS_Pr->SiS_VBType & VB_SIS301LV302LV) && dochiptest) {
11100
11101 temp = (SiS_GetReg(SiS_Pr->SiS_P3d4,0x36) & 0xf0) >> 4;
11102 if(temp == 8) { /* 1400x1050 BIOS (COMPAL) */
11103 delay &= 0x0f;
11104 delay |= 0xb0;
11105 } else if(temp == 6) {
11106 delay &= 0x0f;
11107 delay |= 0xc0;
11108 } else if(temp > 7) { /* 1280x1024 BIOS (which one?) */
11109 delay = 0x35;
11110 }
11111 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x2D,delay);
11112
11113 } else {
11114
11115 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x2D,0xF0,delay);
11116
11117 }
11118
11119 } else { /* LVDS */
11120
11121 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) {
11122 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x2D,0xF0,delay);
11123 } else {
11124 if(IS_SIS650 && (SiS_Pr->SiS_IF_DEF_CH70xx != 0)) {
11125 delay <<= 4;
11126 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x2D,0x0F,delay);
11127 } else {
11128 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x2D,0xF0,delay);
11129 }
11130 }
11131
11132 }
11133
11134}
11135
11136static void
11137SetAntiFlicker(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo,
11138 USHORT ModeNo,USHORT ModeIdIndex)
11139{
11140 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
11141 USHORT index,temp,temp1,romptr=0;
11142
11143 if(SiS_Pr->SiS_TVMode & (TVSetYPbPr750p|TVSetYPbPr525p)) return;
11144
11145 if(ModeNo<=0x13)
11146 index = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].VB_StTVFlickerIndex;
11147 else
11148 index = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].VB_ExtTVFlickerIndex;
11149
11150 temp = GetTVPtrIndex(SiS_Pr);
11151 temp >>= 1; /* 0: NTSC/YPbPr, 1: PAL, 2: HiTV */
11152 temp1 = temp;
11153
11154 if(SiS_Pr->SiS_UseROM && (!(SiS_Pr->SiS_ROMNew))) {
11155 if(HwInfo->jChipType >= SIS_661) {
11156 temp1 = GetOEMTVPtr661(SiS_Pr);
11157 temp1 >>= 1;
11158 romptr = SISGETROMW(0x260);
11159 if(HwInfo->jChipType >= SIS_760) {
11160 romptr = SISGETROMW(0x360);
11161 }
11162 } else if(HwInfo->jChipType >= SIS_330) {
11163 romptr = SISGETROMW(0x192);
11164 } else {
11165 romptr = SISGETROMW(0x112);
11166 }
11167 }
11168
11169 if(romptr) {
11170 temp1 <<= 1;
11171 temp = ROMAddr[romptr + temp1 + index];
11172 } else {
11173 temp = SiS310_TVAntiFlick1[temp][index];
11174 }
11175 temp <<= 4;
11176
11177 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x0A,0x8f,temp); /* index 0A D[6:4] */
11178}
11179
11180static void
11181SetEdgeEnhance(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo,
11182 USHORT ModeNo,USHORT ModeIdIndex)
11183{
11184 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
11185 USHORT index,temp,temp1,romptr=0;
11186
11187 temp = temp1 = GetTVPtrIndex(SiS_Pr) >> 1; /* 0: NTSC/YPbPr, 1: PAL, 2: HiTV */
11188
11189 if(ModeNo <= 0x13)
11190 index = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].VB_StTVEdgeIndex;
11191 else
11192 index = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].VB_ExtTVEdgeIndex;
11193
11194 if(SiS_Pr->SiS_UseROM && (!(SiS_Pr->SiS_ROMNew))) {
11195 if(HwInfo->jChipType >= SIS_661) {
11196 romptr = SISGETROMW(0x26c);
11197 if(HwInfo->jChipType >= SIS_760) {
11198 romptr = SISGETROMW(0x36c);
11199 }
11200 temp1 = GetOEMTVPtr661(SiS_Pr);
11201 temp1 >>= 1;
11202 } else if(HwInfo->jChipType >= SIS_330) {
11203 romptr = SISGETROMW(0x1a4);
11204 } else {
11205 romptr = SISGETROMW(0x124);
11206 }
11207 }
11208
11209 if(romptr) {
11210 temp1 <<= 1;
11211 temp = ROMAddr[romptr + temp1 + index];
11212 } else {
11213 temp = SiS310_TVEdge1[temp][index];
11214 }
11215 temp <<= 5;
11216 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x3A,0x1F,temp); /* index 0A D[7:5] */
11217}
11218
11219static void
11220SetYFilter(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo,
11221 USHORT ModeNo,USHORT ModeIdIndex)
11222{
11223 USHORT index, temp, i, j;
11224
11225 if(ModeNo <= 0x13) {
11226 index = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].VB_StTVYFilterIndex;
11227 } else {
11228 index = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].VB_ExtTVYFilterIndex;
11229 }
11230
11231 temp = GetTVPtrIndex(SiS_Pr) >> 1; /* 0: NTSC/YPbPr, 1: PAL, 2: HiTV */
11232
11233 if(SiS_Pr->SiS_TVMode & TVSetNTSCJ) temp = 1; /* NTSC-J uses PAL */
11234 else if(SiS_Pr->SiS_TVMode & TVSetPALM) temp = 3; /* PAL-M */
11235 else if(SiS_Pr->SiS_TVMode & TVSetPALN) temp = 4; /* PAL-N */
11236 if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) temp = 1; /* HiVision uses PAL */
11237
11238 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) {
11239 for(i=0x35, j=0; i<=0x38; i++, j++) {
11240 SiS_SetReg(SiS_Pr->SiS_Part2Port,i,SiS310_TVYFilter2[temp][index][j]);
11241 }
11242 for(i=0x48; i<=0x4A; i++, j++) {
11243 SiS_SetReg(SiS_Pr->SiS_Part2Port,i,SiS310_TVYFilter2[temp][index][j]);
11244 }
11245 } else {
11246 for(i=0x35, j=0; i<=0x38; i++, j++) {
11247 SiS_SetReg(SiS_Pr->SiS_Part2Port,i,SiS310_TVYFilter1[temp][index][j]);
11248 }
11249 }
11250}
11251
11252static void
11253SetPhaseIncr(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo,
11254 USHORT ModeNo,USHORT ModeIdIndex)
11255{
11256 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
11257 USHORT index,temp,i,j,resinfo,romptr=0;
11258 ULONG lindex;
11259
11260 if(!(SiS_Pr->SiS_VBInfo & SetCRT2ToTV)) return;
11261
11262 /* NTSC-J data not in BIOS, and already set in SetGroup2 */
11263 if(SiS_Pr->SiS_TVMode & TVSetNTSCJ) return;
11264
11265 if((HwInfo->jChipType >= SIS_661) || SiS_Pr->SiS_ROMNew) {
11266 lindex = GetOEMTVPtr661_2_OLD(SiS_Pr) & 0xffff;
11267 lindex <<= 2;
11268 for(j=0, i=0x31; i<=0x34; i++, j++) {
11269 SiS_SetReg(SiS_Pr->SiS_Part2Port,i,SiS661_TVPhase[lindex + j]);
11270 }
11271 return;
11272 }
11273
11274 /* PAL-M, PAL-N not in BIOS, and already set in SetGroup2 */
11275 if(SiS_Pr->SiS_TVMode & (TVSetPALM | TVSetPALN)) return;
11276
11277 if(ModeNo<=0x13) {
11278 resinfo = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_ResInfo;
11279 } else {
11280 resinfo = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_RESINFO;
11281 }
11282
11283 temp = GetTVPtrIndex(SiS_Pr);
11284 /* 0: NTSC Graphics, 1: NTSC Text, 2: PAL Graphics,
11285 * 3: PAL Text, 4: HiTV Graphics 5: HiTV Text
11286 */
11287 if(SiS_Pr->SiS_UseROM) {
11288 romptr = SISGETROMW(0x116);
11289 if(HwInfo->jChipType >= SIS_330) {
11290 romptr = SISGETROMW(0x196);
11291 }
11292 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) {
11293 romptr = SISGETROMW(0x11c);
11294 if(HwInfo->jChipType >= SIS_330) {
11295 romptr = SISGETROMW(0x19c);
11296 }
11297 if((SiS_Pr->SiS_VBInfo & SetInSlaveMode) && (!(SiS_Pr->SiS_TVMode & TVSetTVSimuMode))) {
11298 romptr = SISGETROMW(0x116);
11299 if(HwInfo->jChipType >= SIS_330) {
11300 romptr = SISGETROMW(0x196);
11301 }
11302 }
11303 }
11304 }
11305 if(romptr) {
11306 romptr += (temp << 2);
11307 for(j=0, i=0x31; i<=0x34; i++, j++) {
11308 SiS_SetReg(SiS_Pr->SiS_Part2Port,i,ROMAddr[romptr + j]);
11309 }
11310 } else {
11311 index = temp % 2;
11312 temp >>= 1; /* 0:NTSC, 1:PAL, 2:HiTV */
11313 for(j=0, i=0x31; i<=0x34; i++, j++) {
11314 if(!(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV))
11315 SiS_SetReg(SiS_Pr->SiS_Part2Port,i,SiS310_TVPhaseIncr1[temp][index][j]);
11316 else if((!(SiS_Pr->SiS_VBInfo & SetInSlaveMode)) || (SiS_Pr->SiS_TVMode & TVSetTVSimuMode))
11317 SiS_SetReg(SiS_Pr->SiS_Part2Port,i,SiS310_TVPhaseIncr2[temp][index][j]);
11318 else
11319 SiS_SetReg(SiS_Pr->SiS_Part2Port,i,SiS310_TVPhaseIncr1[temp][index][j]);
11320 }
11321 }
11322
11323 if((SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) && (!(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision))) {
11324 if((!(SiS_Pr->SiS_TVMode & (TVSetPAL | TVSetYPbPr525p | TVSetYPbPr750p))) && (ModeNo > 0x13)) {
11325 if((resinfo == SIS_RI_640x480) ||
11326 (resinfo == SIS_RI_800x600)) {
11327 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x31,0x21);
11328 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x32,0xf0);
11329 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x33,0xf5);
11330 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x34,0x7f);
11331 } else if(resinfo == SIS_RI_1024x768) {
11332 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x31,0x1e);
11333 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x32,0x8b);
11334 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x33,0xfb);
11335 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x34,0x7b);
11336 }
11337 }
11338 }
11339}
11340
11341static void
11342SetDelayComp661(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo, USHORT ModeNo,
11343 USHORT ModeIdIndex, USHORT RTI)
11344{
11345 USHORT delay = 0, romptr = 0, index, lcdpdcindex;
11346 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
11347
11348 if(!(SiS_Pr->SiS_VBInfo & (SetCRT2ToTV | SetCRT2ToLCD | SetCRT2ToLCDA | SetCRT2ToRAMDAC)))
11349 return;
11350
11351 /* 1. New ROM: VGA2 and LCD/LCDA-Pass1:1 */
11352 /* (If a custom mode is used, Pass1:1 is always set; hence we do this:) */
11353
11354 if(SiS_Pr->SiS_ROMNew) {
11355 if((SiS_Pr->SiS_VBInfo & SetCRT2ToRAMDAC) ||
11356 ((SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA)) &&
11357 (SiS_Pr->SiS_LCDInfo & LCDPass11))) {
11358 index = 25;
11359 if(SiS_Pr->UseCustomMode) {
11360 index = SiS_Pr->CSRClock;
11361 } else if(ModeNo > 0x13) {
11362 index = SiS_GetVCLK2Ptr(SiS_Pr,ModeNo,ModeIdIndex,RTI,HwInfo);
11363 index = SiS_Pr->SiS_VCLKData[index].CLOCK;
11364 }
11365 if(index < 25) index = 25;
11366 index = ((index / 25) - 1) << 1;
11367 if((ROMAddr[0x5b] & 0x80) || (SiS_Pr->SiS_VBInfo & (SetCRT2ToRAMDAC | SetCRT2ToLCD))) {
11368 index++;
11369 }
11370 romptr = SISGETROMW(0x104);
11371 delay = ROMAddr[romptr + index];
11372 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToRAMDAC | SetCRT2ToLCD)) {
11373 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x2d,0xf0,((delay >> 1) & 0x0f));
11374 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x35,0x7f,((delay & 0x01) << 7));
11375 } else {
11376 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x2d,0x0f,((delay << 3) & 0xf0));
11377 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x20,0xbf,((delay & 0x01) << 6));
11378 }
11379 return;
11380 }
11381 }
11382
11383 /* 2. Old ROM: VGA2 and LCD/LCDA-Pass 1:1 */
11384
11385 if(SiS_Pr->UseCustomMode) delay = 0x04;
11386 else if(ModeNo <= 0x13) delay = 0x04;
11387 else delay = (SiS_Pr->SiS_RefIndex[RTI].Ext_PDC >> 4);
11388 delay |= (delay << 8);
11389
11390 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) {
11391
11392 /* 3. TV */
11393
11394 index = GetOEMTVPtr661(SiS_Pr);
11395 if(SiS_Pr->SiS_ROMNew) {
11396 romptr = SISGETROMW(0x106);
11397 if(SiS_Pr->SiS_VBType & VB_UMC) romptr += 12;
11398 delay = ROMAddr[romptr + index];
11399 } else {
11400 delay = 0x04;
11401 if(index > 3) delay = 0;
11402 }
11403
11404 } else if(SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA)) {
11405
11406 /* 4. LCD, LCDA (for new ROM only LV and non-Pass 1:1) */
11407
11408 if( (SiS_Pr->SiS_LCDResInfo != Panel_Custom) &&
11409 ((romptr = GetLCDStructPtr661_2(SiS_Pr, HwInfo))) ) {
11410
11411 lcdpdcindex = (SiS_Pr->SiS_VBType & VB_UMC) ? 14 : 12;
11412
11413 /* For LVDS (and sometimes TMDS), the BIOS must know about the correct value */
11414 delay = ROMAddr[romptr + lcdpdcindex + 1]; /* LCD */
11415 delay |= (ROMAddr[romptr + lcdpdcindex] << 8); /* LCDA */
11416
11417 } else {
11418
11419 /* TMDS: Set our own, since BIOS has no idea */
11420 /* (This is done on >=661 only, since <661 is calling this only for LVDS) */
11421 if(!(SiS_Pr->SiS_LCDInfo & LCDPass11)) {
11422 switch(SiS_Pr->SiS_LCDResInfo) {
11423 case Panel_1024x768: delay = 0x0008; break;
11424 case Panel_1280x720: delay = 0x0004; break;
11425 case Panel_1280x768:
11426 case Panel_1280x768_2:delay = 0x0004; break;
11427 case Panel_1280x800:
11428 case Panel_1280x800_2:delay = 0x0004; break; /* Verified for 1280x800 */
11429 case Panel_1280x1024: delay = 0x1e04; break;
11430 case Panel_1400x1050: delay = 0x0004; break;
11431 case Panel_1600x1200: delay = 0x0400; break;
11432 case Panel_1680x1050: delay = 0x0e04; break;
11433 default:
11434 if((SiS_Pr->PanelXRes <= 1024) && (SiS_Pr->PanelYRes <= 768)) {
11435 delay = 0x0008;
11436 } else if((SiS_Pr->PanelXRes == 1280) && (SiS_Pr->PanelYRes == 1024)) {
11437 delay = 0x1e04;
11438 } else if((SiS_Pr->PanelXRes <= 1400) && (SiS_Pr->PanelYRes <= 1050)) {
11439 delay = 0x0004;
11440 } else if((SiS_Pr->PanelXRes <= 1600) && (SiS_Pr->PanelYRes <= 1200)) {
11441 delay = 0x0400;
11442 } else
11443 delay = 0x0e04;
11444 break;
11445 }
11446 }
11447
11448 /* Override by detected or user-set values */
11449 /* (but only if, for some reason, we can't read value from BIOS) */
11450 if((SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) && (SiS_Pr->PDC != -1)) {
11451 delay = SiS_Pr->PDC & 0x1f;
11452 }
11453 if((SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA) && (SiS_Pr->PDCA != -1)) {
11454 delay = (SiS_Pr->PDCA & 0x1f) << 8;
11455 }
11456
11457 }
11458
11459 }
11460
11461 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA) {
11462 delay >>= 8;
11463 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x2d,0x0f,((delay << 3) & 0xf0));
11464 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x20,0xbf,((delay & 0x01) << 6));
11465 } else {
11466 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x2d,0xf0,((delay >> 1) & 0x0f));
11467 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x35,0x7f,((delay & 0x01) << 7));
11468 }
11469}
11470
11471static void
11472SetCRT2SyncDither661(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo, USHORT ModeNo, USHORT RTI)
11473{
11474 USHORT infoflag;
11475 UCHAR temp;
11476
11477 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA)) {
11478
11479 if(ModeNo <= 0x13) {
11480 infoflag = SiS_GetRegByte(SiS_Pr->SiS_P3ca+2);
11481 } else if(SiS_Pr->UseCustomMode) {
11482 infoflag = SiS_Pr->CInfoFlag;
11483 } else {
11484 infoflag = SiS_Pr->SiS_RefIndex[RTI].Ext_InfoFlag;
11485 }
11486
11487 if(!(SiS_Pr->SiS_LCDInfo & LCDPass11)) {
11488 infoflag = SiS_GetReg(SiS_Pr->SiS_P3d4,0x37); /* No longer check D5 */
11489 }
11490
11491 infoflag &= 0xc0;
11492
11493 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) {
11494 temp = (infoflag >> 6) | 0x0c;
11495 if(SiS_Pr->SiS_LCDInfo & LCDRGB18Bit) {
11496 temp ^= 0x04;
11497 if(SiS_Pr->SiS_ModeType >= Mode24Bpp) temp |= 0x10;
11498 }
11499 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x1a,0xe0,temp);
11500 } else {
11501 temp = 0x30;
11502 if(SiS_Pr->SiS_LCDInfo & LCDRGB18Bit) temp = 0x20;
11503 temp |= infoflag;
11504 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x19,0x0f,temp);
11505 temp = 0;
11506 if(SiS_Pr->SiS_LCDInfo & LCDRGB18Bit) {
11507 if(SiS_Pr->SiS_ModeType >= Mode24Bpp) temp |= 0x80;
11508 }
11509 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x1a,0x7f,temp);
11510 }
11511
11512 }
11513}
11514
11515static void
11516SetPanelParms661(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo)
11517{
11518 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
11519 USHORT romptr, temp1, temp2;
11520
11521 if(SiS_Pr->SiS_VBType & (VB_SIS301LV | VB_SIS302LV | VB_SIS302ELV)) {
11522 if(SiS_Pr->LVDSHL != -1) {
11523 SiS_SetRegANDOR(SiS_Pr->SiS_Part4Port,0x24,0xfc,SiS_Pr->LVDSHL);
11524 }
11525 }
11526
11527 if(SiS_Pr->SiS_ROMNew) {
11528
11529 if((romptr = GetLCDStructPtr661_2(SiS_Pr, HwInfo))) {
11530 if(SiS_Pr->SiS_VBType & (VB_SIS301LV | VB_SIS302LV | VB_SIS302ELV)) {
11531 temp1 = (ROMAddr[romptr] & 0x03) | 0x0c;
11532 temp2 = 0xfc;
11533 if(SiS_Pr->LVDSHL != -1) {
11534 temp1 &= 0xfc;
11535 temp2 = 0xf3;
11536 }
11537 SiS_SetRegANDOR(SiS_Pr->SiS_Part4Port,0x24,temp2,temp1);
11538 }
11539 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) {
11540 temp1 = (ROMAddr[romptr + 1] & 0x80) >> 1;
11541 SiS_SetRegANDOR(SiS_Pr->SiS_Part4Port,0x0d,0xbf,temp1);
11542 }
11543 }
11544
11545 }
11546}
11547
11548static void
11549SiS_OEM310Setting(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo,
11550 USHORT ModeNo,USHORT ModeIdIndex,USHORT RRTI)
11551{
11552 if((SiS_Pr->SiS_ROMNew) && (SiS_Pr->SiS_VBType & VB_SISLVDS)) {
11553 SetDelayComp661(SiS_Pr,HwInfo,ModeNo,ModeIdIndex,RRTI);
11554 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA)) {
11555 SetCRT2SyncDither661(SiS_Pr,HwInfo,ModeNo,RRTI);
11556 SetPanelParms661(SiS_Pr,HwInfo);
11557 }
11558 } else {
11559 SetDelayComp(SiS_Pr,HwInfo,ModeNo);
11560 }
11561
11562 if((SiS_Pr->SiS_VBType & VB_SISVB) && (SiS_Pr->SiS_VBInfo & SetCRT2ToTV)) {
11563 SetAntiFlicker(SiS_Pr,HwInfo,ModeNo,ModeIdIndex);
11564 SetPhaseIncr(SiS_Pr,HwInfo,ModeNo,ModeIdIndex);
11565 SetYFilter(SiS_Pr,HwInfo,ModeNo,ModeIdIndex);
11566 if(SiS_Pr->SiS_VBType & VB_SIS301) {
11567 SetEdgeEnhance(SiS_Pr,HwInfo,ModeNo,ModeIdIndex);
11568 }
11569 }
11570}
11571
11572static void
11573SiS_OEM661Setting(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo,
11574 USHORT ModeNo,USHORT ModeIdIndex, USHORT RRTI)
11575{
11576 if(SiS_Pr->SiS_VBType & VB_SISVB) {
11577
11578 SetDelayComp661(SiS_Pr,HwInfo,ModeNo,ModeIdIndex,RRTI);
11579
11580 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA)) {
11581 SetCRT2SyncDither661(SiS_Pr,HwInfo,ModeNo,RRTI);
11582 SetPanelParms661(SiS_Pr,HwInfo);
11583 }
11584
11585 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) {
11586 SetPhaseIncr(SiS_Pr,HwInfo,ModeNo,ModeIdIndex);
11587 SetYFilter(SiS_Pr,HwInfo,ModeNo,ModeIdIndex);
11588 SetAntiFlicker(SiS_Pr,HwInfo,ModeNo,ModeIdIndex);
11589 if(SiS_Pr->SiS_VBType & VB_SIS301) {
11590 SetEdgeEnhance(SiS_Pr,HwInfo,ModeNo,ModeIdIndex);
11591 }
11592 }
11593 }
11594}
11595
11596/* FinalizeLCD
11597 * This finalizes some CRT2 registers for the very panel used.
11598 * If we have a backup if these registers, we use it; otherwise
11599 * we set the register according to most BIOSes. However, this
11600 * function looks quite different in every BIOS, so you better
11601 * pray that we have a backup...
11602 */
11603static void
11604SiS_FinalizeLCD(SiS_Private *SiS_Pr, USHORT ModeNo, USHORT ModeIdIndex,
11605 PSIS_HW_INFO HwInfo)
11606{
11607 USHORT tempcl,tempch,tempbl,tempbh,tempbx,tempax,temp;
11608 USHORT resinfo,modeflag;
11609
11610 if(!(SiS_Pr->SiS_VBType & VB_SIS301LV302LV)) return;
11611 if(SiS_Pr->SiS_ROMNew) return;
11612
11613 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA)) {
11614 if(SiS_Pr->LVDSHL != -1) {
11615 SiS_SetRegANDOR(SiS_Pr->SiS_Part4Port,0x24,0xfc,SiS_Pr->LVDSHL);
11616 }
11617 }
11618
11619 if(SiS_Pr->SiS_LCDResInfo == Panel_Custom) return;
11620 if(SiS_Pr->UseCustomMode) return;
11621
11622 switch(SiS_Pr->SiS_CustomT) {
11623 case CUT_COMPAQ1280:
11624 case CUT_COMPAQ12802:
11625 case CUT_CLEVO1400:
11626 case CUT_CLEVO14002:
11627 return;
11628 }
11629
11630 if(ModeNo <= 0x13) {
11631 resinfo = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_ResInfo;
11632 modeflag = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_ModeFlag;
11633 } else {
11634 resinfo = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_RESINFO;
11635 modeflag = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_ModeFlag;
11636 }
11637
11638 if(IS_SIS650) {
11639 if(!(SiS_GetReg(SiS_Pr->SiS_P3d4, 0x5f) & 0xf0)) {
11640 if(SiS_Pr->SiS_CustomT == CUT_CLEVO1024) {
11641 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x1e,0x02);
11642 } else {
11643 SiS_SetRegOR(SiS_Pr->SiS_Part1Port,0x1e,0x03);
11644 }
11645 }
11646 }
11647
11648 if(SiS_Pr->SiS_CustomT == CUT_CLEVO1024) {
11649 if(SiS_Pr->SiS_LCDResInfo == Panel_1024x768) {
11650 /* Maybe all panels? */
11651 if(SiS_Pr->LVDSHL == -1) {
11652 SiS_SetRegANDOR(SiS_Pr->SiS_Part4Port,0x24,0xfc,0x01);
11653 }
11654 return;
11655 }
11656 }
11657
11658 if(SiS_Pr->SiS_CustomT == CUT_CLEVO10242) {
11659 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA)) {
11660 if(SiS_Pr->SiS_LCDResInfo == Panel_1024x768) {
11661 if(SiS_Pr->LVDSHL == -1) {
11662 /* Maybe all panels? */
11663 SiS_SetRegANDOR(SiS_Pr->SiS_Part4Port,0x24,0xfc,0x01);
11664 }
11665 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA) {
11666 tempch = SiS_GetReg(SiS_Pr->SiS_P3d4,0x36) >> 4;
11667 if(tempch == 3) {
11668 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x18,0x02);
11669 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1b,0x25);
11670 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1c,0x00);
11671 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1d,0x1b);
11672 }
11673 }
11674 return;
11675 }
11676 }
11677 }
11678
11679 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToLCD | SetCRT2ToLCDA)) {
11680 if(SiS_Pr->SiS_LCDResInfo == Panel_1024x768) {
11681 if(SiS_Pr->SiS_VBType & (VB_SIS302LV | VB_SIS302ELV)) {
11682 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x2a,0x00);
11683#ifdef SET_EMI
11684 SiS_SetRegAND(SiS_Pr->SiS_Part4Port,0x30,0x0c);
11685#endif
11686 SiS_SetReg(SiS_Pr->SiS_Part4Port,0x34,0x10);
11687 }
11688 } else if(SiS_Pr->SiS_LCDResInfo == Panel_1280x1024) {
11689 if(SiS_Pr->LVDSHL == -1) {
11690 /* Maybe ACER only? */
11691 SiS_SetRegANDOR(SiS_Pr->SiS_Part4Port,0x24,0xfc,0x01);
11692 }
11693 }
11694 tempch = SiS_GetReg(SiS_Pr->SiS_P3d4,0x36) >> 4;
11695 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCDA) {
11696 if(SiS_Pr->SiS_LCDResInfo == Panel_1400x1050) {
11697 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1f,0x76);
11698 } else if(SiS_Pr->SiS_LCDResInfo == Panel_1024x768) {
11699 if(tempch == 0x03) {
11700 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x18,0x02);
11701 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1b,0x25);
11702 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1c,0x00);
11703 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1d,0x1b);
11704 }
11705 if((SiS_Pr->Backup == TRUE) && (SiS_Pr->Backup_Mode == ModeNo)) {
11706 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x14,SiS_Pr->Backup_14);
11707 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x15,SiS_Pr->Backup_15);
11708 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x16,SiS_Pr->Backup_16);
11709 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x17,SiS_Pr->Backup_17);
11710 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x18,SiS_Pr->Backup_18);
11711 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x19,SiS_Pr->Backup_19);
11712 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1a,SiS_Pr->Backup_1a);
11713 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1b,SiS_Pr->Backup_1b);
11714 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1c,SiS_Pr->Backup_1c);
11715 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1d,SiS_Pr->Backup_1d);
11716 } else if(!(SiS_Pr->SiS_LCDInfo & DontExpandLCD)) { /* 1.10.8w */
11717 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x14,0x90);
11718 if(ModeNo <= 0x13) {
11719 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x18,0x11);
11720 if((resinfo == 0) || (resinfo == 2)) return;
11721 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x18,0x18);
11722 if((resinfo == 1) || (resinfo == 3)) return;
11723 }
11724 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x18,0x02);
11725 if((ModeNo > 0x13) && (resinfo == SIS_RI_1024x768)) {
11726 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x18,0x02); /* 1.10.7u */
11727#if 0
11728 tempbx = 806; /* 0x326 */ /* other older BIOSes */
11729 tempbx--;
11730 temp = tempbx & 0xff;
11731 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1b,temp);
11732 temp = (tempbx >> 8) & 0x03;
11733 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x1d,0xf8,temp);
11734#endif
11735 }
11736 } else if(ModeNo <= 0x13) {
11737 if(ModeNo <= 1) {
11738 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x18,0x70);
11739 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x19,0xff);
11740 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1b,0x48);
11741 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1d,0x12);
11742 }
11743 if(!(modeflag & HalfDCLK)) {
11744 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x14,0x20);
11745 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x15,0x1a);
11746 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x16,0x28);
11747 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x17,0x00);
11748 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x18,0x4c);
11749 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x19,0xdc);
11750 if(ModeNo == 0x12) {
11751 switch(tempch) {
11752 case 0:
11753 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x18,0x95);
11754 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x19,0xdc);
11755 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1a,0x10);
11756 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1b,0x95);
11757 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1c,0x48);
11758 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1d,0x12);
11759 break;
11760 case 2:
11761 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x18,0x95);
11762 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1b,0x48);
11763 break;
11764 case 3:
11765 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x1b,0x95);
11766 break;
11767 }
11768 }
11769 }
11770 }
11771 }
11772 } else {
11773 tempcl = tempbh = SiS_GetReg(SiS_Pr->SiS_Part2Port,0x01);
11774 tempcl &= 0x0f;
11775 tempbh &= 0x70;
11776 tempbh >>= 4;
11777 tempbl = SiS_GetReg(SiS_Pr->SiS_Part2Port,0x04);
11778 tempbx = (tempbh << 8) | tempbl;
11779 if(SiS_Pr->SiS_LCDResInfo == Panel_1024x768) {
11780 if((resinfo == SIS_RI_1024x768) || (!(SiS_Pr->SiS_LCDInfo & DontExpandLCD))) {
11781 if(SiS_Pr->SiS_SetFlag & LCDVESATiming) {
11782 tempbx = 770;
11783 } else {
11784 if(tempbx > 770) tempbx = 770;
11785 if(SiS_Pr->SiS_VGAVDE < 600) {
11786 tempax = 768 - SiS_Pr->SiS_VGAVDE;
11787 tempax >>= 4; /* 1.10.7w; 1.10.6s: 3; */
11788 if(SiS_Pr->SiS_VGAVDE <= 480) tempax >>= 4; /* 1.10.7w; 1.10.6s: < 480; >>=1; */
11789 tempbx -= tempax;
11790 }
11791 }
11792 } else return;
11793 }
11794 temp = tempbx & 0xff;
11795 SiS_SetReg(SiS_Pr->SiS_Part2Port,0x04,temp);
11796 temp = ((tempbx & 0xff00) >> 4) | tempcl;
11797 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x01,0x80,temp);
11798 }
11799 }
11800}
11801
11802#endif
11803
11804/* ================= SiS 300 O.E.M. ================== */
11805
11806#ifdef SIS300
11807
11808static void
11809SetOEMLCDData2(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo,
11810 USHORT ModeNo,USHORT ModeIdIndex, USHORT RefTabIndex)
11811{
11812 USHORT crt2crtc=0, modeflag, myindex=0;
11813 UCHAR temp;
11814 int i;
11815
11816 if(ModeNo <= 0x13) {
11817 modeflag = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_ModeFlag;
11818 crt2crtc = SiS_Pr->SiS_SModeIDTable[ModeIdIndex].St_CRT2CRTC;
11819 } else {
11820 modeflag = SiS_Pr->SiS_EModeIDTable[ModeIdIndex].Ext_ModeFlag;
11821 crt2crtc = SiS_Pr->SiS_RefIndex[RefTabIndex].Ext_CRT2CRTC;
11822 }
11823
11824 crt2crtc &= 0x3f;
11825
11826 if(SiS_Pr->SiS_CustomT == CUT_BARCO1024) {
11827 SiS_SetRegAND(SiS_Pr->SiS_Part1Port,0x13,0xdf);
11828 }
11829
11830 if(SiS_Pr->SiS_CustomT == CUT_BARCO1366) {
11831 if(modeflag & HalfDCLK) myindex = 1;
11832
11833 if(SiS_Pr->SiS_SetFlag & LowModeTests) {
11834 for(i=0; i<7; i++) {
11835 if(barco_p1[myindex][crt2crtc][i][0]) {
11836 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,
11837 barco_p1[myindex][crt2crtc][i][0],
11838 barco_p1[myindex][crt2crtc][i][2],
11839 barco_p1[myindex][crt2crtc][i][1]);
11840 }
11841 }
11842 }
11843 temp = SiS_GetReg(SiS_Pr->SiS_Part1Port,0x00);
11844 if(temp & 0x80) {
11845 temp = SiS_GetReg(SiS_Pr->SiS_Part1Port,0x18);
11846 temp++;
11847 SiS_SetReg(SiS_Pr->SiS_Part1Port,0x18,temp);
11848 }
11849 }
11850}
11851
11852static USHORT
11853GetOEMLCDPtr(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo, int Flag)
11854{
11855 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
11856 USHORT tempbx=0,romptr=0;
11857 UCHAR customtable300[] = {
11858 0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,
11859 0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff
11860 };
11861 UCHAR customtable630[] = {
11862 0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,
11863 0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff
11864 };
11865
11866 if(HwInfo->jChipType == SIS_300) {
11867
11868 tempbx = SiS_GetReg(SiS_Pr->SiS_P3d4,0x36) & 0x0f;
11869 if(SiS_Pr->SiS_VBType & VB_SIS301) tempbx &= 0x07;
11870 tempbx -= 2;
11871 if(!(SiS_Pr->SiS_SetFlag & LCDVESATiming)) tempbx += 4;
11872 if(SiS_Pr->SiS_LCDResInfo == Panel_1024x768) {
11873 if(SiS_Pr->SiS_LCDInfo & DontExpandLCD) tempbx += 3;
11874 }
11875 if(SiS_Pr->SiS_UseROM) {
11876 if(ROMAddr[0x235] & 0x80) {
11877 tempbx = SiS_Pr->SiS_LCDTypeInfo;
11878 if(Flag) {
11879 romptr = SISGETROMW(0x255);
11880 if(romptr) tempbx = ROMAddr[romptr + SiS_Pr->SiS_LCDTypeInfo];
11881 else tempbx = customtable300[SiS_Pr->SiS_LCDTypeInfo];
11882 if(tempbx == 0xFF) return 0xFFFF;
11883 }
11884 tempbx <<= 1;
11885 if(!(SiS_Pr->SiS_SetFlag & LCDVESATiming)) tempbx++;
11886 }
11887 }
11888
11889 } else {
11890
11891 if(Flag) {
11892 if(SiS_Pr->SiS_UseROM) {
11893 romptr = SISGETROMW(0x255);
11894 if(romptr) tempbx = ROMAddr[romptr + SiS_Pr->SiS_LCDTypeInfo];
11895 else tempbx = 0xff;
11896 } else {
11897 tempbx = customtable630[SiS_Pr->SiS_LCDTypeInfo];
11898 }
11899 if(tempbx == 0xFF) return 0xFFFF;
11900 tempbx <<= 2;
11901 if(SiS_Pr->SiS_VBInfo & SetInSlaveMode) tempbx += 2;
11902 if(SiS_Pr->SiS_LCDInfo & DontExpandLCD) tempbx++;
11903 return tempbx;
11904 }
11905 tempbx = SiS_Pr->SiS_LCDTypeInfo << 2;
11906 if(SiS_Pr->SiS_VBInfo & SetInSlaveMode) tempbx += 2;
11907 if(SiS_Pr->SiS_LCDInfo & DontExpandLCD) tempbx++;
11908
11909 }
11910
11911 return tempbx;
11912}
11913
11914static void
11915SetOEMLCDDelay(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo,
11916 USHORT ModeNo,USHORT ModeIdIndex)
11917{
11918 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
11919 USHORT index,temp,romptr=0;
11920
11921 if(SiS_Pr->SiS_LCDResInfo == Panel_Custom) return;
11922
11923 if(SiS_Pr->SiS_UseROM) {
11924 if(!(ROMAddr[0x237] & 0x01)) return;
11925 if(!(ROMAddr[0x237] & 0x02)) return;
11926 romptr = SISGETROMW(0x24b);
11927 }
11928
11929 /* The Panel Compensation Delay should be set according to tables
11930 * here. Unfortunately, various BIOS versions don't case about
11931 * a uniform way using eg. ROM byte 0x220, but use different
11932 * hard coded delays (0x04, 0x20, 0x18) in SetGroup1().
11933 * Thus we don't set this if the user select a custom pdc or if
11934 * we otherwise detected a valid pdc.
11935 */
11936 if(SiS_Pr->PDC != -1) return;
11937
11938 temp = GetOEMLCDPtr(SiS_Pr,HwInfo, 0);
11939
11940 if(SiS_Pr->UseCustomMode)
11941 index = 0;
11942 else
11943 index = SiS_Pr->SiS_VBModeIDTable[ModeIdIndex].VB_LCDDelayIndex;
11944
11945 if(HwInfo->jChipType != SIS_300) {
11946 if(romptr) {
11947 romptr += (temp * 2);
11948 romptr = SISGETROMW(romptr);
11949 romptr += index;
11950 temp = ROMAddr[romptr];
11951 } else {
11952 if(SiS_Pr->SiS_VBType & VB_SISVB) {
11953 temp = SiS300_OEMLCDDelay2[temp][index];
11954 } else {
11955 temp = SiS300_OEMLCDDelay3[temp][index];
11956 }
11957 }
11958 } else {
11959 if(SiS_Pr->SiS_UseROM && (ROMAddr[0x235] & 0x80)) {
11960 if(romptr) {
11961 romptr += (temp * 2);
11962 romptr = SISGETROMW(romptr);
11963 romptr += index;
11964 temp = ROMAddr[romptr];
11965 } else {
11966 temp = SiS300_OEMLCDDelay5[temp][index];
11967 }
11968 } else {
11969 if(SiS_Pr->SiS_UseROM) {
11970 romptr = ROMAddr[0x249] | (ROMAddr[0x24a] << 8);
11971 if(romptr) {
11972 romptr += (temp * 2);
11973 romptr = SISGETROMW(romptr);
11974 romptr += index;
11975 temp = ROMAddr[romptr];
11976 } else {
11977 temp = SiS300_OEMLCDDelay4[temp][index];
11978 }
11979 } else {
11980 temp = SiS300_OEMLCDDelay4[temp][index];
11981 }
11982 }
11983 }
11984 temp &= 0x3c;
11985 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x13,~0x3C,temp); /* index 0A D[6:4] */
11986}
11987
11988static void
11989SetOEMLCDData(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo,
11990 USHORT ModeNo,USHORT ModeIdIndex)
11991{
11992#if 0 /* Unfinished; Data table missing */
11993 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
11994 USHORT index,temp;
11995
11996 if((SiS_Pr->SiS_UseROM) {
11997 if(!(ROMAddr[0x237] & 0x01)) return;
11998 if(!(ROMAddr[0x237] & 0x04)) return;
11999 /* No rom pointer in BIOS header! */
12000 }
12001
12002 temp = GetOEMLCDPtr(SiS_Pr,HwInfo, 1);
12003 if(temp = 0xFFFF) return;
12004
12005 index = SiS_Pr->SiS_VBModeIDTable[ModeIdIndex]._VB_LCDHIndex;
12006 for(i=0x14, j=0; i<=0x17; i++, j++) {
12007 SiS_SetReg(SiS_Pr->SiS_Part1Port,i,SiS300_LCDHData[temp][index][j]);
12008 }
12009 SiS_SetRegANDOR(SiS_SiS_Part1Port,0x1a, 0xf8, (SiS300_LCDHData[temp][index][j] & 0x07));
12010
12011 index = SiS_Pr->SiS_VBModeIDTable[ModeIdIndex]._VB_LCDVIndex;
12012 SiS_SetReg(SiS_SiS_Part1Port,0x18, SiS300_LCDVData[temp][index][0]);
12013 SiS_SetRegANDOR(SiS_SiS_Part1Port,0x19, 0xF0, SiS300_LCDVData[temp][index][1]);
12014 SiS_SetRegANDOR(SiS_SiS_Part1Port,0x1A, 0xC7, (SiS300_LCDVData[temp][index][2] & 0x38));
12015 for(i=0x1b, j=3; i<=0x1d; i++, j++) {
12016 SiS_SetReg(SiS_Pr->SiS_Part1Port,i,SiS300_LCDVData[temp][index][j]);
12017 }
12018#endif
12019}
12020
12021static USHORT
12022GetOEMTVPtr(SiS_Private *SiS_Pr)
12023{
12024 USHORT index;
12025
12026 index = 0;
12027 if(!(SiS_Pr->SiS_VBInfo & SetInSlaveMode)) index += 4;
12028 if(SiS_Pr->SiS_VBType & VB_SISVB) {
12029 if(SiS_Pr->SiS_VBInfo & SetCRT2ToSCART) index += 2;
12030 else if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) index += 3;
12031 else if(SiS_Pr->SiS_TVMode & TVSetPAL) index += 1;
12032 } else {
12033 if(SiS_Pr->SiS_TVMode & TVSetCHOverScan) index += 2;
12034 if(SiS_Pr->SiS_TVMode & TVSetPAL) index += 1;
12035 }
12036 return index;
12037}
12038
12039static void
12040SetOEMTVDelay(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo,
12041 USHORT ModeNo,USHORT ModeIdIndex)
12042{
12043 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
12044 USHORT index,temp,romptr=0;
12045
12046 if(SiS_Pr->SiS_UseROM) {
12047 if(!(ROMAddr[0x238] & 0x01)) return;
12048 if(!(ROMAddr[0x238] & 0x02)) return;
12049 romptr = SISGETROMW(0x241);
12050 }
12051
12052 temp = GetOEMTVPtr(SiS_Pr);
12053
12054 index = SiS_Pr->SiS_VBModeIDTable[ModeIdIndex].VB_TVDelayIndex;
12055
12056 if(romptr) {
12057 romptr += (temp * 2);
12058 romptr = SISGETROMW(romptr);
12059 romptr += index;
12060 temp = ROMAddr[romptr];
12061 } else {
12062 if(SiS_Pr->SiS_VBType & VB_SISVB) {
12063 temp = SiS300_OEMTVDelay301[temp][index];
12064 } else {
12065 temp = SiS300_OEMTVDelayLVDS[temp][index];
12066 }
12067 }
12068 temp &= 0x3c;
12069 SiS_SetRegANDOR(SiS_Pr->SiS_Part1Port,0x13,~0x3C,temp);
12070}
12071
12072static void
12073SetOEMAntiFlicker(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo,
12074 USHORT ModeNo, USHORT ModeIdIndex)
12075{
12076 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
12077 USHORT index,temp,romptr=0;
12078
12079 if(SiS_Pr->SiS_UseROM) {
12080 if(!(ROMAddr[0x238] & 0x01)) return;
12081 if(!(ROMAddr[0x238] & 0x04)) return;
12082 romptr = SISGETROMW(0x243);
12083 }
12084
12085 temp = GetOEMTVPtr(SiS_Pr);
12086
12087 index = SiS_Pr->SiS_VBModeIDTable[ModeIdIndex].VB_TVFlickerIndex;
12088
12089 if(romptr) {
12090 romptr += (temp * 2);
12091 romptr = SISGETROMW(romptr);
12092 romptr += index;
12093 temp = ROMAddr[romptr];
12094 } else {
12095 temp = SiS300_OEMTVFlicker[temp][index];
12096 }
12097 temp &= 0x70;
12098 SiS_SetRegANDOR(SiS_Pr->SiS_Part2Port,0x0A,0x8F,temp);
12099}
12100
12101static void
12102SetOEMPhaseIncr(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo,
12103 USHORT ModeNo,USHORT ModeIdIndex)
12104{
12105 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
12106 USHORT index,i,j,temp,romptr=0;
12107
12108 if(SiS_Pr->SiS_VBInfo & SetCRT2ToHiVision) return;
12109
12110 if(SiS_Pr->SiS_TVMode & (TVSetNTSC1024 | TVSetNTSCJ | TVSetPALM | TVSetPALN)) return;
12111
12112 if(SiS_Pr->SiS_UseROM) {
12113 if(!(ROMAddr[0x238] & 0x01)) return;
12114 if(!(ROMAddr[0x238] & 0x08)) return;
12115 romptr = SISGETROMW(0x245);
12116 }
12117
12118 temp = GetOEMTVPtr(SiS_Pr);
12119
12120 index = SiS_Pr->SiS_VBModeIDTable[ModeIdIndex].VB_TVPhaseIndex;
12121
12122 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) {
12123 for(i=0x31, j=0; i<=0x34; i++, j++) {
12124 SiS_SetReg(SiS_Pr->SiS_Part2Port,i,SiS300_Phase2[temp][index][j]);
12125 }
12126 } else {
12127 if(romptr) {
12128 romptr += (temp * 2);
12129 romptr = SISGETROMW(romptr);
12130 romptr += (index * 4);
12131 for(i=0x31, j=0; i<=0x34; i++, j++) {
12132 SiS_SetReg(SiS_Pr->SiS_Part2Port,i,ROMAddr[romptr + j]);
12133 }
12134 } else {
12135 for(i=0x31, j=0; i<=0x34; i++, j++) {
12136 SiS_SetReg(SiS_Pr->SiS_Part2Port,i,SiS300_Phase1[temp][index][j]);
12137 }
12138 }
12139 }
12140}
12141
12142static void
12143SetOEMYFilter(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo,
12144 USHORT ModeNo,USHORT ModeIdIndex)
12145{
12146 UCHAR *ROMAddr = HwInfo->pjVirtualRomBase;
12147 USHORT index,temp,i,j,romptr=0;
12148
12149 if(SiS_Pr->SiS_VBInfo & (SetCRT2ToSCART | SetCRT2ToHiVision | SetCRT2ToYPbPr525750)) return;
12150
12151 if(SiS_Pr->SiS_UseROM) {
12152 if(!(ROMAddr[0x238] & 0x01)) return;
12153 if(!(ROMAddr[0x238] & 0x10)) return;
12154 romptr = SISGETROMW(0x247);
12155 }
12156
12157 temp = GetOEMTVPtr(SiS_Pr);
12158
12159 if(SiS_Pr->SiS_TVMode & TVSetPALM) temp = 8;
12160 else if(SiS_Pr->SiS_TVMode & TVSetPALN) temp = 9;
12161 /* NTSCJ uses NTSC filters */
12162
12163 index = SiS_Pr->SiS_VBModeIDTable[ModeIdIndex].VB_TVYFilterIndex;
12164
12165 if(SiS_Pr->SiS_VBType & VB_SIS301BLV302BLV) {
12166 for(i=0x35, j=0; i<=0x38; i++, j++) {
12167 SiS_SetReg(SiS_Pr->SiS_Part2Port,i,SiS300_Filter2[temp][index][j]);
12168 }
12169 for(i=0x48; i<=0x4A; i++, j++) {
12170 SiS_SetReg(SiS_Pr->SiS_Part2Port,i,SiS300_Filter2[temp][index][j]);
12171 }
12172 } else {
12173 if((romptr) && (!(SiS_Pr->SiS_TVMode & (TVSetPALM|TVSetPALN)))) {
12174 romptr += (temp * 2);
12175 romptr = SISGETROMW(romptr);
12176 romptr += (index * 4);
12177 for(i=0x35, j=0; i<=0x38; i++, j++) {
12178 SiS_SetReg(SiS_Pr->SiS_Part2Port,i,ROMAddr[romptr + j]);
12179 }
12180 } else {
12181 for(i=0x35, j=0; i<=0x38; i++, j++) {
12182 SiS_SetReg(SiS_Pr->SiS_Part2Port,i,SiS300_Filter1[temp][index][j]);
12183 }
12184 }
12185 }
12186}
12187
12188static USHORT
12189SiS_SearchVBModeID(SiS_Private *SiS_Pr, USHORT *ModeNo)
12190{
12191 USHORT ModeIdIndex;
12192 UCHAR VGAINFO = SiS_Pr->SiS_VGAINFO;
12193
12194 if(*ModeNo <= 5) *ModeNo |= 1;
12195
12196 for(ModeIdIndex=0; ; ModeIdIndex++) {
12197 if(SiS_Pr->SiS_VBModeIDTable[ModeIdIndex].ModeID == *ModeNo) break;
12198 if(SiS_Pr->SiS_VBModeIDTable[ModeIdIndex].ModeID == 0xFF) return 0;
12199 }
12200
12201 if(*ModeNo != 0x07) {
12202 if(*ModeNo > 0x03) return ModeIdIndex;
12203 if(VGAINFO & 0x80) return ModeIdIndex;
12204 ModeIdIndex++;
12205 }
12206
12207 if(VGAINFO & 0x10) ModeIdIndex++; /* 400 lines */
12208 /* else 350 lines */
12209 return ModeIdIndex;
12210}
12211
12212static void
12213SiS_OEM300Setting(SiS_Private *SiS_Pr, PSIS_HW_INFO HwInfo,
12214 USHORT ModeNo, USHORT ModeIdIndex, USHORT RefTableIndex)
12215{
12216 USHORT OEMModeIdIndex=0;
12217
12218 if(!SiS_Pr->UseCustomMode) {
12219 OEMModeIdIndex = SiS_SearchVBModeID(SiS_Pr,&ModeNo);
12220 if(!(OEMModeIdIndex)) return;
12221 }
12222
12223 if(SiS_Pr->SiS_VBInfo & SetCRT2ToLCD) {
12224 SetOEMLCDDelay(SiS_Pr, HwInfo, ModeNo, OEMModeIdIndex);
12225 if(SiS_Pr->SiS_IF_DEF_LVDS == 1) {
12226 SetOEMLCDData(SiS_Pr, HwInfo, ModeNo, OEMModeIdIndex);
12227 }
12228 }
12229 if(SiS_Pr->UseCustomMode) return;
12230 if(SiS_Pr->SiS_VBInfo & SetCRT2ToTV) {
12231 SetOEMTVDelay(SiS_Pr, HwInfo, ModeNo,OEMModeIdIndex);
12232 if(SiS_Pr->SiS_VBType & VB_SISVB) {
12233 SetOEMAntiFlicker(SiS_Pr, HwInfo, ModeNo, OEMModeIdIndex);
12234 SetOEMPhaseIncr(SiS_Pr, HwInfo, ModeNo, OEMModeIdIndex);
12235 SetOEMYFilter(SiS_Pr, HwInfo, ModeNo, OEMModeIdIndex);
12236 }
12237 }
12238}
12239#endif
12240