xref: /haiku/src/add-ons/accelerants/nvidia/engine/nv_brooktreetv.c (revision c5fe5f2d061e6ee95d7d3280030a5c0fad451c95)
112566cbdSRudolf Cornelissen /*
212566cbdSRudolf Cornelissen 	Author:
3b2459715SRudolf Cornelissen 	Rudolf Cornelissen 4/2002-10/2005
4fb6cadf1Sshatty */
5fb6cadf1Sshatty 
6fb6cadf1Sshatty #define MODULE_BIT 0x00100000
7fb6cadf1Sshatty 
8fb6cadf1Sshatty #include "nv_std.h"
9fb6cadf1Sshatty 
1012566cbdSRudolf Cornelissen #define PRADR	0x88
1112566cbdSRudolf Cornelissen #define SCADR	0x8a
1212566cbdSRudolf Cornelissen #define WR		0x00
1312566cbdSRudolf Cornelissen #define RD		0x01
1412566cbdSRudolf Cornelissen 
1521f6ecabSRudolf Cornelissen //fixme: remove rubbish..
1621f6ecabSRudolf Cornelissen enum
1721f6ecabSRudolf Cornelissen {	// TVoutput mode to set
18b2459715SRudolf Cornelissen 	NOT_SUPPORTED = 0,
1921f6ecabSRudolf Cornelissen 	NTSC640_TST,
2021f6ecabSRudolf Cornelissen 	NTSC640,
2121f6ecabSRudolf Cornelissen 	NTSC800,
2221f6ecabSRudolf Cornelissen 	PAL800_TST,
2321f6ecabSRudolf Cornelissen 	PAL640,
2421f6ecabSRudolf Cornelissen 	PAL800,
2521f6ecabSRudolf Cornelissen 	VGA,
2621f6ecabSRudolf Cornelissen 	EXIT,
2721f6ecabSRudolf Cornelissen 	NTSC720,
2821f6ecabSRudolf Cornelissen 	PAL720,
2921f6ecabSRudolf Cornelissen 	NTSC640_OS,
3021f6ecabSRudolf Cornelissen 	PAL800_OS
3121f6ecabSRudolf Cornelissen };
3221f6ecabSRudolf Cornelissen 
339a7218c5SRudolf Cornelissen /* Dirk Thierbach's Macro setup for registers 0xda-0xfe */
349a7218c5SRudolf Cornelissen static uint8 BtNtscMacro0 [] = {
359a7218c5SRudolf Cornelissen   0x0f,0xfc,0x20,0xd0,0x6f,0x0f,0x00,0x00,0x0c,0xf3,0x09,
369a7218c5SRudolf Cornelissen   0xbd,0x67,0xb5,0x90,0xb2,0x7d,0x00,0x00};
379a7218c5SRudolf Cornelissen static uint8 BtNtscMacro1 [] = {
389a7218c5SRudolf Cornelissen   0x0f,0xfc,0x20,0xd0,0x6f,0x0f,0x00,0x00,0x0c,0xf3,0x09,
399a7218c5SRudolf Cornelissen   0xbd,0x67,0xb5,0x90,0xb2,0x7d,0x63,0x00};
409a7218c5SRudolf Cornelissen static uint8 BtNtscMacro2 [] = {
419a7218c5SRudolf Cornelissen   0x0f,0xfc,0x20,0xd0,0x6f,0x0f,0x00,0x00,0x0c,0xf3,0x09,
429a7218c5SRudolf Cornelissen   0xbd,0x6c,0x31,0x92,0x32,0xdd,0xe3,0x00};
439a7218c5SRudolf Cornelissen static uint8 BtNtscMacro3 [] = {
449a7218c5SRudolf Cornelissen   0x0f,0xfc,0x20,0xd0,0x6f,0x0f,0x00,0x00,0x0c,0xf3,0x09,
459a7218c5SRudolf Cornelissen   0xbd,0x66,0xb5,0x90,0xb2,0x7d,0xe3,0x00};
469a7218c5SRudolf Cornelissen 
479a7218c5SRudolf Cornelissen static uint8 BtPalMacro0 [] = {
489a7218c5SRudolf Cornelissen   0x05,0x57,0x20,0x40,0x6e,0x7e,0xf4,0x51,0x0f,0xf1,0x05,
499a7218c5SRudolf Cornelissen   0xd3,0x78,0xa2,0x25,0x54,0xa5,0x00,0x00};
509a7218c5SRudolf Cornelissen static uint8 BtPalMacro1 [] = {
519a7218c5SRudolf Cornelissen   0x05,0x57,0x20,0x40,0x6e,0x7e,0xf4,0x51,0x0f,0xf1,0x05,
529a7218c5SRudolf Cornelissen   0xd3,0x78,0xa2,0x25,0x54,0xa5,0x63,0x00};
539a7218c5SRudolf Cornelissen 
549a7218c5SRudolf Cornelissen static uint8 BT_set_macro (int std, int mode)
559a7218c5SRudolf Cornelissen {
569a7218c5SRudolf Cornelissen 	uint8 stat;
579a7218c5SRudolf Cornelissen 	uint8 buffer[21];
589a7218c5SRudolf Cornelissen 
599a7218c5SRudolf Cornelissen 	LOG(4,("Brooktree: Setting Macro\n"));
609a7218c5SRudolf Cornelissen 
619a7218c5SRudolf Cornelissen 	if ((std < 0) | (std > 1) | (mode < 0) | (mode > 3))
629a7218c5SRudolf Cornelissen 	{
639a7218c5SRudolf Cornelissen 		LOG(4,("Brooktree: Non existing mode or standard selected, aborting.\n"));
649a7218c5SRudolf Cornelissen 		return 0x80;
659a7218c5SRudolf Cornelissen 	}
669a7218c5SRudolf Cornelissen 
679a7218c5SRudolf Cornelissen 	switch (std)
689a7218c5SRudolf Cornelissen 	{
699a7218c5SRudolf Cornelissen 	case 0:
709a7218c5SRudolf Cornelissen 		/* NTSC */
719a7218c5SRudolf Cornelissen 		switch (mode)
729a7218c5SRudolf Cornelissen 		{
739a7218c5SRudolf Cornelissen 		case 0:
749a7218c5SRudolf Cornelissen 			/* disabled */
759a7218c5SRudolf Cornelissen 			memcpy(&buffer[2], &BtNtscMacro0, 19);
769a7218c5SRudolf Cornelissen 			break;
779a7218c5SRudolf Cornelissen 		case 1:
789a7218c5SRudolf Cornelissen 			/* enabled mode 1 */
799a7218c5SRudolf Cornelissen 			memcpy(&buffer[2], &BtNtscMacro1, 19);
809a7218c5SRudolf Cornelissen 			break;
819a7218c5SRudolf Cornelissen 		case 2:
829a7218c5SRudolf Cornelissen 			/* enabled mode 2 */
839a7218c5SRudolf Cornelissen 			memcpy(&buffer[2], &BtNtscMacro2, 19);
849a7218c5SRudolf Cornelissen 			break;
859a7218c5SRudolf Cornelissen 		case 3:
869a7218c5SRudolf Cornelissen 			/* enabled mode 3 */
879a7218c5SRudolf Cornelissen 			memcpy(&buffer[2], &BtNtscMacro3, 19);
889a7218c5SRudolf Cornelissen 			break;
899a7218c5SRudolf Cornelissen 		}
909a7218c5SRudolf Cornelissen 		break;
919a7218c5SRudolf Cornelissen 	case 1:
929a7218c5SRudolf Cornelissen 		/* PAL */
939a7218c5SRudolf Cornelissen 		switch (mode)
949a7218c5SRudolf Cornelissen 		{
959a7218c5SRudolf Cornelissen 		case 0:
969a7218c5SRudolf Cornelissen 			/* disabled */
979a7218c5SRudolf Cornelissen 			memcpy(&buffer[2], &BtPalMacro0, 19);
989a7218c5SRudolf Cornelissen 			break;
999a7218c5SRudolf Cornelissen 		case 1:
1009a7218c5SRudolf Cornelissen 		case 2:
1019a7218c5SRudolf Cornelissen 		case 3:
1029a7218c5SRudolf Cornelissen 			/* enabled */
1039a7218c5SRudolf Cornelissen 			memcpy(&buffer[2], &BtPalMacro1, 19);
1049a7218c5SRudolf Cornelissen 			break;
1059a7218c5SRudolf Cornelissen 		}
1069a7218c5SRudolf Cornelissen 		break;
1079a7218c5SRudolf Cornelissen 	}
1089a7218c5SRudolf Cornelissen 
1099a7218c5SRudolf Cornelissen 	buffer[0] = si->ps.tv_encoder.adress + WR;
1109a7218c5SRudolf Cornelissen 	/* select first register to write to */
1119a7218c5SRudolf Cornelissen 	buffer[1] = 0xda;
1129a7218c5SRudolf Cornelissen 
1139a7218c5SRudolf Cornelissen 	/* reset status */
1149a7218c5SRudolf Cornelissen 	i2c_flag_error (-1);
1159a7218c5SRudolf Cornelissen 
1169a7218c5SRudolf Cornelissen 	i2c_bstart(si->ps.tv_encoder.bus);
1179a7218c5SRudolf Cornelissen 	i2c_writebuffer(si->ps.tv_encoder.bus, buffer, sizeof(buffer));
1189a7218c5SRudolf Cornelissen 	i2c_bstop(si->ps.tv_encoder.bus);
1199a7218c5SRudolf Cornelissen 	/* log on errors */
1209a7218c5SRudolf Cornelissen 	stat = i2c_flag_error(0);
1219a7218c5SRudolf Cornelissen 	if (stat)
1229a7218c5SRudolf Cornelissen 		LOG(4,("Brooktree: I2C errors occurred while setting Macro\n"));
1239a7218c5SRudolf Cornelissen 
1249a7218c5SRudolf Cornelissen 	return stat;
1259a7218c5SRudolf Cornelissen }//end BT_set_macro.
1269a7218c5SRudolf Cornelissen 
12712566cbdSRudolf Cornelissen /*
12812566cbdSRudolf Cornelissen  see if a (possible) BT/CX chip resides at the given adress.
12912566cbdSRudolf Cornelissen  Return zero if no errors occurred.
13012566cbdSRudolf Cornelissen */
13112566cbdSRudolf Cornelissen static uint8 BT_check (uint8 bus, uint8 adress)
13212566cbdSRudolf Cornelissen {
1330ece4905SRudolf Cornelissen 	uint8 buffer[3];
13412566cbdSRudolf Cornelissen 
1350ece4905SRudolf Cornelissen 	buffer[0] = adress + WR;
13612566cbdSRudolf Cornelissen 	/* set ESTATUS at b'00'; and enable bt chip-outputs
13712566cbdSRudolf Cornelissen 	 * WARNING:
13812566cbdSRudolf Cornelissen 	 * If bit0 = 0 is issued below (EN_OUT = disabled), the BT will lock SDA
13912566cbdSRudolf Cornelissen 	 * after writing adress $A0 (setting EN_XCLK)!!!
1409a7218c5SRudolf Cornelissen 	 * Until a reboot the corresponding I2C bus will be inacessable then!!! */
1410ece4905SRudolf Cornelissen 	buffer[1] = 0xc4;
14212566cbdSRudolf Cornelissen 	/* fixme: if testimage 'was' active txbuffer[3] should become 0x05...
14312566cbdSRudolf Cornelissen 	 * (currently this cannot be detected in a 'foolproof' way so don't touch...) */
14412566cbdSRudolf Cornelissen 	/* (ESTATUS b'0x' means: RX ID and VERSION info later..) */
1450ece4905SRudolf Cornelissen 	buffer[2] = 0x01;
1460ece4905SRudolf Cornelissen 
1470ece4905SRudolf Cornelissen 	/* reset status */
1480ece4905SRudolf Cornelissen 	i2c_flag_error (-1);
1490ece4905SRudolf Cornelissen 
1500ece4905SRudolf Cornelissen 	/* do check */
1510ece4905SRudolf Cornelissen 	i2c_bstart(bus);
1520ece4905SRudolf Cornelissen 	i2c_writebuffer(bus, buffer, sizeof(buffer));
15312566cbdSRudolf Cornelissen 	i2c_bstop(bus);
15412566cbdSRudolf Cornelissen 	return i2c_flag_error(0);
15512566cbdSRudolf Cornelissen }
15612566cbdSRudolf Cornelissen 
15712566cbdSRudolf Cornelissen /* identify chiptype */
15812566cbdSRudolf Cornelissen static uint8 BT_read_type (void)
15912566cbdSRudolf Cornelissen {
1600ece4905SRudolf Cornelissen 	uint8 id, type, stat;
1610ece4905SRudolf Cornelissen 	uint8 buffer[3];
1620ece4905SRudolf Cornelissen 
1630ece4905SRudolf Cornelissen 	/* Make sure a CX (Conexant) chip (if this turns out to be there) is set to
1640ece4905SRudolf Cornelissen 	 * BT-compatibility mode! (This command will do nothing on a BT chip...) */
1650ece4905SRudolf Cornelissen 	buffer[0] = si->ps.tv_encoder.adress + WR;
1660ece4905SRudolf Cornelissen 	/* select CX reg. for BT-compatible readback, video still off */
1670ece4905SRudolf Cornelissen 	buffer[1] = 0x6c;
1680ece4905SRudolf Cornelissen 	/* set it up */
1690ece4905SRudolf Cornelissen 	buffer[2] = 0x02;
17012566cbdSRudolf Cornelissen 
17112566cbdSRudolf Cornelissen 	/* reset status */
17212566cbdSRudolf Cornelissen 	i2c_flag_error (-1);
17312566cbdSRudolf Cornelissen 
17412566cbdSRudolf Cornelissen 	i2c_bstart(si->ps.tv_encoder.bus);
1750ece4905SRudolf Cornelissen 	i2c_writebuffer(si->ps.tv_encoder.bus, buffer, sizeof(buffer));
17612566cbdSRudolf Cornelissen 	i2c_bstop(si->ps.tv_encoder.bus);
17712566cbdSRudolf Cornelissen 	/* abort on errors */
17812566cbdSRudolf Cornelissen 	stat = i2c_flag_error(0);
17912566cbdSRudolf Cornelissen 	if (stat) return stat;
18012566cbdSRudolf Cornelissen 
18112566cbdSRudolf Cornelissen 	/* Do actual readtype command */
18212566cbdSRudolf Cornelissen 	i2c_bstart(si->ps.tv_encoder.bus);
1839a7218c5SRudolf Cornelissen 	/* issue I2C read command */
18412566cbdSRudolf Cornelissen 	i2c_writebyte(si->ps.tv_encoder.bus, si->ps.tv_encoder.adress + RD);
18512566cbdSRudolf Cornelissen 	/* receive 1 byte;
1869a7218c5SRudolf Cornelissen 	 * ACK level to TX after last byte to RX should be 1 (= NACK) (see I2C spec). */
18712566cbdSRudolf Cornelissen 	/* note:
18812566cbdSRudolf Cornelissen 	 * While the BT's don't care, CX chips will block the SDA line if
18912566cbdSRudolf Cornelissen 	 * an ACK gets sent! */
19012566cbdSRudolf Cornelissen 	id = i2c_readbyte(si->ps.tv_encoder.bus, true);
19112566cbdSRudolf Cornelissen 	i2c_bstop(si->ps.tv_encoder.bus);
19212566cbdSRudolf Cornelissen 	/* abort on errors */
19312566cbdSRudolf Cornelissen 	stat = i2c_flag_error(0);
19412566cbdSRudolf Cornelissen 	if (stat) return stat;
19512566cbdSRudolf Cornelissen 
1960ece4905SRudolf Cornelissen 	/* check type to be supported one */
1970ece4905SRudolf Cornelissen 	type = (id & 0xe0) >> 5;
1980ece4905SRudolf Cornelissen 	if (type > 3)
1990ece4905SRudolf Cornelissen 	{
2000ece4905SRudolf Cornelissen 		LOG(4,("Brooktree: Found unsupported encoder type %d, aborting.\n", type));
2010ece4905SRudolf Cornelissen 		return 0x80;
2020ece4905SRudolf Cornelissen 	}
2030ece4905SRudolf Cornelissen 
20412566cbdSRudolf Cornelissen 	/* inform driver about TV encoder found */
20512566cbdSRudolf Cornelissen 	si->ps.tvout = true;
2060ece4905SRudolf Cornelissen 	si->ps.tv_encoder.type = BT868 + type;
20712566cbdSRudolf Cornelissen 	si->ps.tv_encoder.version = id & 0x1f;
20812566cbdSRudolf Cornelissen 
20912566cbdSRudolf Cornelissen 	return stat;
21012566cbdSRudolf Cornelissen }
21112566cbdSRudolf Cornelissen 
21212566cbdSRudolf Cornelissen bool BT_probe()
21312566cbdSRudolf Cornelissen {
21412566cbdSRudolf Cornelissen 	bool btfound = false;
21512566cbdSRudolf Cornelissen 
2169a7218c5SRudolf Cornelissen 	LOG(4,("Brooktree: Checking I2C bus(ses) for first possible TV encoder...\n"));
21712566cbdSRudolf Cornelissen 	if (si->ps.i2c_bus0)
21812566cbdSRudolf Cornelissen 	{
21912566cbdSRudolf Cornelissen 		/* try primary adress on bus 0 */
22012566cbdSRudolf Cornelissen 		if (!BT_check(0, PRADR))
22112566cbdSRudolf Cornelissen 		{
22212566cbdSRudolf Cornelissen 			btfound = true;
22312566cbdSRudolf Cornelissen 			si->ps.tv_encoder.adress = PRADR;
22412566cbdSRudolf Cornelissen 			si->ps.tv_encoder.bus = 0;
22512566cbdSRudolf Cornelissen 		}
22612566cbdSRudolf Cornelissen 		else
22712566cbdSRudolf Cornelissen 		{
22812566cbdSRudolf Cornelissen 			/* try secondary adress on bus 0 */
22912566cbdSRudolf Cornelissen 			if (!BT_check(0, SCADR))
23012566cbdSRudolf Cornelissen 			{
23112566cbdSRudolf Cornelissen 				btfound = true;
23212566cbdSRudolf Cornelissen 				si->ps.tv_encoder.adress = SCADR;
23312566cbdSRudolf Cornelissen 				si->ps.tv_encoder.bus = 0;
23412566cbdSRudolf Cornelissen 			}
23512566cbdSRudolf Cornelissen 		}
23612566cbdSRudolf Cornelissen 	}
23712566cbdSRudolf Cornelissen 
23812566cbdSRudolf Cornelissen 	if (si->ps.i2c_bus1 && !btfound)
23912566cbdSRudolf Cornelissen 	{
24012566cbdSRudolf Cornelissen 		/* try primary adress on bus 1 */
24112566cbdSRudolf Cornelissen 		if (!BT_check(1, PRADR))
24212566cbdSRudolf Cornelissen 		{
24312566cbdSRudolf Cornelissen 			btfound = true;
24412566cbdSRudolf Cornelissen 			si->ps.tv_encoder.adress = PRADR;
24512566cbdSRudolf Cornelissen 			si->ps.tv_encoder.bus = 1;
24612566cbdSRudolf Cornelissen 		}
24712566cbdSRudolf Cornelissen 		else
24812566cbdSRudolf Cornelissen 		{
24912566cbdSRudolf Cornelissen 			/* try secondary adress on bus 1 */
25012566cbdSRudolf Cornelissen 			if (!BT_check(1, SCADR))
25112566cbdSRudolf Cornelissen 			{
25212566cbdSRudolf Cornelissen 				btfound = true;
25312566cbdSRudolf Cornelissen 				si->ps.tv_encoder.adress = SCADR;
25412566cbdSRudolf Cornelissen 				si->ps.tv_encoder.bus = 1;
25512566cbdSRudolf Cornelissen 			}
25612566cbdSRudolf Cornelissen 		}
25712566cbdSRudolf Cornelissen 	}
25812566cbdSRudolf Cornelissen 
25912566cbdSRudolf Cornelissen 	/* identify exact TV encoder type */
26012566cbdSRudolf Cornelissen 	if (btfound)
26112566cbdSRudolf Cornelissen 	{
26212566cbdSRudolf Cornelissen 		/* if errors are found, retry */
26312566cbdSRudolf Cornelissen 		/* note:
26412566cbdSRudolf Cornelissen 		 * NACK: occurs on some ASUS V7700 GeForce cards!
26512566cbdSRudolf Cornelissen 		 * (apparantly the video-in chip or another chip resides at 'BT' adresses
26612566cbdSRudolf Cornelissen 		 * there..) */
26712566cbdSRudolf Cornelissen 		uint8 stat;
26812566cbdSRudolf Cornelissen 		uint8 cnt = 0;
26912566cbdSRudolf Cornelissen 		while ((stat = BT_read_type()) && (cnt < 3))
27012566cbdSRudolf Cornelissen 		{
2710ece4905SRudolf Cornelissen 			/* don't retry on unsupported chiptype */
2720ece4905SRudolf Cornelissen 			if (stat == 0x80)
2730ece4905SRudolf Cornelissen 			{
2740ece4905SRudolf Cornelissen 				btfound = 0;
2750ece4905SRudolf Cornelissen 				break;
2760ece4905SRudolf Cornelissen 			}
27712566cbdSRudolf Cornelissen 			cnt++;
27812566cbdSRudolf Cornelissen 		}
2790ece4905SRudolf Cornelissen 		if (stat & 0x7f)
28012566cbdSRudolf Cornelissen 		{
2819a7218c5SRudolf Cornelissen 			LOG(4,("Brooktree: Too much errors occurred, aborting.\n"));
28212566cbdSRudolf Cornelissen 			btfound = 0;
28312566cbdSRudolf Cornelissen 		}
28412566cbdSRudolf Cornelissen 	}
28512566cbdSRudolf Cornelissen 
28612566cbdSRudolf Cornelissen 	if (btfound)
28712566cbdSRudolf Cornelissen 		LOG(4,("Brooktree: Found TV encoder on bus %d, adress $%02x\n",
28812566cbdSRudolf Cornelissen 			si->ps.tv_encoder.bus, si->ps.tv_encoder.adress));
28912566cbdSRudolf Cornelissen 	else
29012566cbdSRudolf Cornelissen 		LOG(4,("Brooktree: No TV encoder Found\n"));
29112566cbdSRudolf Cornelissen 
29212566cbdSRudolf Cornelissen 	return btfound;
29312566cbdSRudolf Cornelissen }
29412566cbdSRudolf Cornelissen 
29577497c49SRudolf Cornelissen static uint8 BT_init_PAL640()
296fb6cadf1Sshatty {
29777497c49SRudolf Cornelissen 	uint8 stat;
298fb6cadf1Sshatty 
29977497c49SRudolf Cornelissen 	uint8 buffer[35];
300fb6cadf1Sshatty 
3019a7218c5SRudolf Cornelissen 	LOG(4,("Brooktree: Setting PAL 640x480 desktop mode\n"));
3029a7218c5SRudolf Cornelissen 
3039a7218c5SRudolf Cornelissen 	buffer[0] = si->ps.tv_encoder.adress + WR;	//issue I2C write command
30477497c49SRudolf Cornelissen 	buffer[1] = 0x76;			//select first bt register to write to
30577497c49SRudolf Cornelissen 	buffer[2] = 0x60;
30677497c49SRudolf Cornelissen 	buffer[3] = 0x80;
30777497c49SRudolf Cornelissen 	buffer[4] = 0x8a;
30877497c49SRudolf Cornelissen 	buffer[5] = 0xa6;
30977497c49SRudolf Cornelissen 	buffer[6] = 0x68;
31077497c49SRudolf Cornelissen 	buffer[7] = 0xc1;
31177497c49SRudolf Cornelissen 	buffer[8] = 0x2e;
31277497c49SRudolf Cornelissen 	buffer[9] = 0xf2;
31377497c49SRudolf Cornelissen 	buffer[10] = 0x27;
31477497c49SRudolf Cornelissen 	buffer[11] = 0x00;
31577497c49SRudolf Cornelissen 	buffer[12] = 0xb0;
31677497c49SRudolf Cornelissen 	buffer[13] = 0x0a;
31777497c49SRudolf Cornelissen 	buffer[14] = 0x0b;
31877497c49SRudolf Cornelissen 	buffer[15] = 0x71;
31977497c49SRudolf Cornelissen 	buffer[16] = 0x5a;
32077497c49SRudolf Cornelissen 	buffer[17] = 0xe0;
32177497c49SRudolf Cornelissen 	buffer[18] = 0x36;
32277497c49SRudolf Cornelissen 	buffer[19] = 0x00;
32377497c49SRudolf Cornelissen 	buffer[20] = 0x50;
32477497c49SRudolf Cornelissen 	buffer[21] = 0x72;
32577497c49SRudolf Cornelissen 	buffer[22] = 0x1c;
32677497c49SRudolf Cornelissen 	buffer[23] = 0x8d;		//chip-pin CLKI is pixel clock (only non-default here!)
32777497c49SRudolf Cornelissen 	buffer[24] = 0x24;
32877497c49SRudolf Cornelissen 	buffer[25] = 0xf0;
32977497c49SRudolf Cornelissen 	buffer[26] = 0x58;
33077497c49SRudolf Cornelissen 	buffer[27] = 0x81;
33177497c49SRudolf Cornelissen 	buffer[28] = 0x49;
33277497c49SRudolf Cornelissen 	buffer[29] = 0x8c;
33377497c49SRudolf Cornelissen 	buffer[30] = 0x0c;
33477497c49SRudolf Cornelissen 	buffer[31] = 0x8c;
33577497c49SRudolf Cornelissen 	buffer[32] = 0x79;
33677497c49SRudolf Cornelissen 	buffer[33] = 0x26;
33777497c49SRudolf Cornelissen 	buffer[34] = 0x00;
33877497c49SRudolf Cornelissen 
33977497c49SRudolf Cornelissen 	/* reset status */
34077497c49SRudolf Cornelissen 	i2c_flag_error (-1);
34177497c49SRudolf Cornelissen 
34277497c49SRudolf Cornelissen 	i2c_bstart(si->ps.tv_encoder.bus);
34377497c49SRudolf Cornelissen 	i2c_writebuffer(si->ps.tv_encoder.bus, buffer, sizeof(buffer));
34477497c49SRudolf Cornelissen 	i2c_bstop(si->ps.tv_encoder.bus);
34577497c49SRudolf Cornelissen 	/* log on errors */
34677497c49SRudolf Cornelissen 	stat = i2c_flag_error(0);
34777497c49SRudolf Cornelissen 	if (stat)
34877497c49SRudolf Cornelissen 		LOG(4,("Brooktree: I2C errors occurred while setting mode PAL640\n"));
34977497c49SRudolf Cornelissen 
35077497c49SRudolf Cornelissen 	return stat;
35177497c49SRudolf Cornelissen }//end BT_init_PAL640.
35277497c49SRudolf Cornelissen 
35377497c49SRudolf Cornelissen static uint8 BT_init_PAL800()
354fb6cadf1Sshatty {
35577497c49SRudolf Cornelissen 	uint8 stat;
356fb6cadf1Sshatty 
35777497c49SRudolf Cornelissen 	uint8 buffer[35];
358fb6cadf1Sshatty 
3599a7218c5SRudolf Cornelissen 	LOG(4,("Brooktree: Setting PAL 800x600 desktop mode\n"));
3609a7218c5SRudolf Cornelissen 
3619a7218c5SRudolf Cornelissen 	buffer[0] = si->ps.tv_encoder.adress + WR;	//issue I2C write command
36277497c49SRudolf Cornelissen 	buffer[1] = 0x76;			//select first bt register to write to
36377497c49SRudolf Cornelissen 	buffer[2] = 0x00;
36477497c49SRudolf Cornelissen 	buffer[3] = 0x20;
36577497c49SRudolf Cornelissen 	buffer[4] = 0xaa;
36677497c49SRudolf Cornelissen 	buffer[5] = 0xca;
36777497c49SRudolf Cornelissen 	buffer[6] = 0x9a;
36877497c49SRudolf Cornelissen 	buffer[7] = 0x0d;
36977497c49SRudolf Cornelissen 	buffer[8] = 0x29;
37077497c49SRudolf Cornelissen 	buffer[9] = 0xfc;
37177497c49SRudolf Cornelissen 	buffer[10] = 0x39;
37277497c49SRudolf Cornelissen 	buffer[11] = 0x00;
37377497c49SRudolf Cornelissen 	buffer[12] = 0xc0;
37477497c49SRudolf Cornelissen 	buffer[13] = 0x8c;
37577497c49SRudolf Cornelissen 	buffer[14] = 0x03;
37677497c49SRudolf Cornelissen 	buffer[15] = 0xee;
37777497c49SRudolf Cornelissen 	buffer[16] = 0x5f;
37877497c49SRudolf Cornelissen 	buffer[17] = 0x58;
37977497c49SRudolf Cornelissen 	buffer[18] = 0x3a;
38077497c49SRudolf Cornelissen 	buffer[19] = 0x66;
38177497c49SRudolf Cornelissen 	buffer[20] = 0x96;
38277497c49SRudolf Cornelissen 	buffer[21] = 0x00;
38377497c49SRudolf Cornelissen 	buffer[22] = 0x00;
38477497c49SRudolf Cornelissen 	buffer[23] = 0x90;		//chip-pin CLKI is pixel clock (only non-default here!)
38577497c49SRudolf Cornelissen 	buffer[24] = 0x24;
38677497c49SRudolf Cornelissen 	buffer[25] = 0xf0;
38777497c49SRudolf Cornelissen 	buffer[26] = 0x57;
38877497c49SRudolf Cornelissen 	buffer[27] = 0x80;
38977497c49SRudolf Cornelissen 	buffer[28] = 0x48;
39077497c49SRudolf Cornelissen 	buffer[29] = 0x8c;
39177497c49SRudolf Cornelissen 	buffer[30] = 0x18;
39277497c49SRudolf Cornelissen 	buffer[31] = 0x28;
39377497c49SRudolf Cornelissen 	buffer[32] = 0x87;
39477497c49SRudolf Cornelissen 	buffer[33] = 0x1f;
39577497c49SRudolf Cornelissen 	buffer[34] = 0x00;
39677497c49SRudolf Cornelissen 
39777497c49SRudolf Cornelissen 	/* reset status */
39877497c49SRudolf Cornelissen 	i2c_flag_error (-1);
39977497c49SRudolf Cornelissen 
40077497c49SRudolf Cornelissen 	i2c_bstart(si->ps.tv_encoder.bus);
40177497c49SRudolf Cornelissen 	i2c_writebuffer(si->ps.tv_encoder.bus, buffer, sizeof(buffer));
40277497c49SRudolf Cornelissen 	i2c_bstop(si->ps.tv_encoder.bus);
40377497c49SRudolf Cornelissen 	/* log on errors */
40477497c49SRudolf Cornelissen 	stat = i2c_flag_error(0);
40577497c49SRudolf Cornelissen 	if (stat)
40677497c49SRudolf Cornelissen 		LOG(4,("Brooktree: I2C errors occurred while setting mode PAL800\n"));
40777497c49SRudolf Cornelissen 
40877497c49SRudolf Cornelissen 	return stat;
40977497c49SRudolf Cornelissen }//end BT_init_PAL800.
41077497c49SRudolf Cornelissen 
41177497c49SRudolf Cornelissen static uint8 BT_init_NTSC640()
412fb6cadf1Sshatty {
41377497c49SRudolf Cornelissen 	uint8 stat;
414fb6cadf1Sshatty 
41577497c49SRudolf Cornelissen 	uint8 buffer[35];
41677497c49SRudolf Cornelissen 
4179a7218c5SRudolf Cornelissen 	LOG(4,("Brooktree: Setting NTSC 640x480 desktop mode\n"));
4189a7218c5SRudolf Cornelissen 
4199a7218c5SRudolf Cornelissen 	buffer[0] = si->ps.tv_encoder.adress + WR;	//issue I2C write command
42077497c49SRudolf Cornelissen 	buffer[1] = 0x76;			//select first bt register to write to
42177497c49SRudolf Cornelissen 	buffer[2] = 0x00;
42277497c49SRudolf Cornelissen 	buffer[3] = 0x80;
42377497c49SRudolf Cornelissen 	buffer[4] = 0x84;
42477497c49SRudolf Cornelissen 	buffer[5] = 0x96;
42577497c49SRudolf Cornelissen 	buffer[6] = 0x60;
42677497c49SRudolf Cornelissen 	buffer[7] = 0x7d;
42777497c49SRudolf Cornelissen 	buffer[8] = 0x22;
42877497c49SRudolf Cornelissen 	buffer[9] = 0xd4;
42977497c49SRudolf Cornelissen 	buffer[10] = 0x27;
43077497c49SRudolf Cornelissen 	buffer[11] = 0x00;
43177497c49SRudolf Cornelissen 	buffer[12] = 0x10;
43277497c49SRudolf Cornelissen 	buffer[13] = 0x7e;
43377497c49SRudolf Cornelissen 	buffer[14] = 0x03;
43477497c49SRudolf Cornelissen 	buffer[15] = 0x58;
43577497c49SRudolf Cornelissen 	buffer[16] = 0x4b;
43677497c49SRudolf Cornelissen 	buffer[17] = 0xe0;
43777497c49SRudolf Cornelissen 	buffer[18] = 0x36;
43877497c49SRudolf Cornelissen 	buffer[19] = 0x92;
43977497c49SRudolf Cornelissen 	buffer[20] = 0x54;
44077497c49SRudolf Cornelissen 	buffer[21] = 0x0e;
44177497c49SRudolf Cornelissen 	buffer[22] = 0x88;
44277497c49SRudolf Cornelissen 	buffer[23] = 0x8c;		//chip-pin CLKI is pixel clock (only non-default here!)
44377497c49SRudolf Cornelissen 	buffer[24] = 0x0a;
44477497c49SRudolf Cornelissen 	buffer[25] = 0xe5;
44577497c49SRudolf Cornelissen 	buffer[26] = 0x76;
44677497c49SRudolf Cornelissen 	buffer[27] = 0x79;
44777497c49SRudolf Cornelissen 	buffer[28] = 0x44;
44877497c49SRudolf Cornelissen 	buffer[29] = 0x85;
44977497c49SRudolf Cornelissen 	buffer[30] = 0x00;
45077497c49SRudolf Cornelissen 	buffer[31] = 0x00;
45177497c49SRudolf Cornelissen 	buffer[32] = 0x80;
45277497c49SRudolf Cornelissen 	buffer[33] = 0x20;
45377497c49SRudolf Cornelissen 	buffer[34] = 0x00;
45477497c49SRudolf Cornelissen 
45577497c49SRudolf Cornelissen 	/* reset status */
45677497c49SRudolf Cornelissen 	i2c_flag_error (-1);
45777497c49SRudolf Cornelissen 
45877497c49SRudolf Cornelissen 	i2c_bstart(si->ps.tv_encoder.bus);
45977497c49SRudolf Cornelissen 	i2c_writebuffer(si->ps.tv_encoder.bus, buffer, sizeof(buffer));
46077497c49SRudolf Cornelissen 	i2c_bstop(si->ps.tv_encoder.bus);
46177497c49SRudolf Cornelissen 	/* log on errors */
46277497c49SRudolf Cornelissen 	stat = i2c_flag_error(0);
46377497c49SRudolf Cornelissen 	if (stat)
46477497c49SRudolf Cornelissen 		LOG(4,("Brooktree: I2C errors occurred while setting mode NTSC640\n"));
46577497c49SRudolf Cornelissen 
46677497c49SRudolf Cornelissen 	return stat;
46777497c49SRudolf Cornelissen }//end BT_init_NTSC640.
46877497c49SRudolf Cornelissen 
46977497c49SRudolf Cornelissen static uint8 BT_init_NTSC800()
470fb6cadf1Sshatty {
47177497c49SRudolf Cornelissen 	uint8 stat;
472fb6cadf1Sshatty 
47377497c49SRudolf Cornelissen 	uint8 buffer[35];
474fb6cadf1Sshatty 
4759a7218c5SRudolf Cornelissen 	LOG(4,("Brooktree: Setting NTSC 800x600 desktop mode\n"));
4769a7218c5SRudolf Cornelissen 
4779a7218c5SRudolf Cornelissen 	buffer[0] = si->ps.tv_encoder.adress + WR;	//issue I2C write command
47877497c49SRudolf Cornelissen 	buffer[1] = 0x76;			//select first bt register to write to
47977497c49SRudolf Cornelissen 	buffer[2] = 0xa0;
48077497c49SRudolf Cornelissen 	buffer[3] = 0x20;
48177497c49SRudolf Cornelissen 	buffer[4] = 0xb6;
48277497c49SRudolf Cornelissen 	buffer[5] = 0xce;
48377497c49SRudolf Cornelissen 	buffer[6] = 0x84;
48477497c49SRudolf Cornelissen 	buffer[7] = 0x55;
48577497c49SRudolf Cornelissen 	buffer[8] = 0x20;
48677497c49SRudolf Cornelissen 	buffer[9] = 0xd8;
48777497c49SRudolf Cornelissen 	buffer[10] = 0x39;
48877497c49SRudolf Cornelissen 	buffer[11] = 0x00;
48977497c49SRudolf Cornelissen 	buffer[12] = 0x70;
49077497c49SRudolf Cornelissen 	buffer[13] = 0x42;
49177497c49SRudolf Cornelissen 	buffer[14] = 0x03;
49277497c49SRudolf Cornelissen 	buffer[15] = 0xdf;
49377497c49SRudolf Cornelissen 	buffer[16] = 0x56;
49477497c49SRudolf Cornelissen 	buffer[17] = 0x58;
49577497c49SRudolf Cornelissen 	buffer[18] = 0x3a;
49677497c49SRudolf Cornelissen 	buffer[19] = 0xcd;
49777497c49SRudolf Cornelissen 	buffer[20] = 0x9c;
49877497c49SRudolf Cornelissen 	buffer[21] = 0x14;
49977497c49SRudolf Cornelissen 	buffer[22] = 0x3b;
50077497c49SRudolf Cornelissen 	buffer[23] = 0x91;		//chip-pin CLKI is pixel clock (only non-default here!)
50177497c49SRudolf Cornelissen 	buffer[24] = 0x0a;
50277497c49SRudolf Cornelissen 	buffer[25] = 0xe5;
50377497c49SRudolf Cornelissen 	buffer[26] = 0x74;
50477497c49SRudolf Cornelissen 	buffer[27] = 0x77;
50577497c49SRudolf Cornelissen 	buffer[28] = 0x43;
50677497c49SRudolf Cornelissen 	buffer[29] = 0x85;
50777497c49SRudolf Cornelissen 	buffer[30] = 0xba;
50877497c49SRudolf Cornelissen 	buffer[31] = 0xe8;
50977497c49SRudolf Cornelissen 	buffer[32] = 0xa2;
51077497c49SRudolf Cornelissen 	buffer[33] = 0x17;
51177497c49SRudolf Cornelissen 	buffer[34] = 0x00;
51277497c49SRudolf Cornelissen 
51377497c49SRudolf Cornelissen 	/* reset status */
51477497c49SRudolf Cornelissen 	i2c_flag_error (-1);
51577497c49SRudolf Cornelissen 
51677497c49SRudolf Cornelissen 	i2c_bstart(si->ps.tv_encoder.bus);
51777497c49SRudolf Cornelissen 	i2c_writebuffer(si->ps.tv_encoder.bus, buffer, sizeof(buffer));
51877497c49SRudolf Cornelissen 	i2c_bstop(si->ps.tv_encoder.bus);
51977497c49SRudolf Cornelissen 	/* log on errors */
52077497c49SRudolf Cornelissen 	stat = i2c_flag_error(0);
52177497c49SRudolf Cornelissen 	if (stat)
52277497c49SRudolf Cornelissen 		LOG(4,("Brooktree: I2C errors occurred while setting mode PAL800\n"));
52377497c49SRudolf Cornelissen 
52477497c49SRudolf Cornelissen 	return stat;
52577497c49SRudolf Cornelissen }//end BT_init_NTSC800.
52677497c49SRudolf Cornelissen 
52777497c49SRudolf Cornelissen static uint8 BT_init_PAL720()
528fb6cadf1Sshatty {
52977497c49SRudolf Cornelissen 	uint8 stat;
53077497c49SRudolf Cornelissen 
53177497c49SRudolf Cornelissen 	uint8 buffer[35];
53277497c49SRudolf Cornelissen 
5339a7218c5SRudolf Cornelissen 	LOG(4,("Brooktree: Setting PAL 720x576 overscanning DVD mode\n"));
5349a7218c5SRudolf Cornelissen 
5359a7218c5SRudolf Cornelissen 	buffer[0] = si->ps.tv_encoder.adress + WR;	//issue I2C write command
53677497c49SRudolf Cornelissen 	buffer[1] = 0x76;			//select first bt register to write to
53777497c49SRudolf Cornelissen 	buffer[2] = 0xf0;
53877497c49SRudolf Cornelissen 	buffer[3] = 0xd0;
53977497c49SRudolf Cornelissen 	buffer[4] = 0x82;
54077497c49SRudolf Cornelissen 	buffer[5] = 0x9c;
54177497c49SRudolf Cornelissen 	buffer[6] = 0x5a;
54277497c49SRudolf Cornelissen 	buffer[7] = 0x31;
54377497c49SRudolf Cornelissen 	buffer[8] = 0x16;
54477497c49SRudolf Cornelissen 	buffer[9] = 0x22;
54577497c49SRudolf Cornelissen 	buffer[10] = 0xa6;
54677497c49SRudolf Cornelissen 	buffer[11] = 0x00;
54777497c49SRudolf Cornelissen 	buffer[12] = 0x78;
54877497c49SRudolf Cornelissen 	buffer[13] = 0x93;
54977497c49SRudolf Cornelissen 	buffer[14] = 0x03;
55077497c49SRudolf Cornelissen 	buffer[15] = 0x71;
55177497c49SRudolf Cornelissen 	buffer[16] = 0x2a;
55277497c49SRudolf Cornelissen 	buffer[17] = 0x40;
55377497c49SRudolf Cornelissen 	buffer[18] = 0x0a;
55477497c49SRudolf Cornelissen 	buffer[19] = 0x00;
55577497c49SRudolf Cornelissen 	buffer[20] = 0x50;
55677497c49SRudolf Cornelissen 	buffer[21] = 0x55;
55777497c49SRudolf Cornelissen 	buffer[22] = 0x55;
55877497c49SRudolf Cornelissen 	buffer[23] = 0x8c;		//chip-pin CLKI is pixel clock (only non-default here!)
55977497c49SRudolf Cornelissen 	buffer[24] = 0x24;
56077497c49SRudolf Cornelissen 	buffer[25] = 0xf0;
56177497c49SRudolf Cornelissen 	buffer[26] = 0x59;
56277497c49SRudolf Cornelissen 	buffer[27] = 0x82;
56377497c49SRudolf Cornelissen 	buffer[28] = 0x49;
56477497c49SRudolf Cornelissen 	buffer[29] = 0x8c;
56577497c49SRudolf Cornelissen 	buffer[30] = 0x8e;
56677497c49SRudolf Cornelissen 	buffer[31] = 0xb0;
56777497c49SRudolf Cornelissen 	buffer[32] = 0xe6;
56877497c49SRudolf Cornelissen 	buffer[33] = 0x28;
56977497c49SRudolf Cornelissen 	buffer[34] = 0x00;
57077497c49SRudolf Cornelissen 
57177497c49SRudolf Cornelissen 	/* reset status */
57277497c49SRudolf Cornelissen 	i2c_flag_error (-1);
57377497c49SRudolf Cornelissen 
57477497c49SRudolf Cornelissen 	i2c_bstart(si->ps.tv_encoder.bus);
57577497c49SRudolf Cornelissen 	i2c_writebuffer(si->ps.tv_encoder.bus, buffer, sizeof(buffer));
57677497c49SRudolf Cornelissen 	i2c_bstop(si->ps.tv_encoder.bus);
57777497c49SRudolf Cornelissen 	/* log on errors */
57877497c49SRudolf Cornelissen 	stat = i2c_flag_error(0);
57977497c49SRudolf Cornelissen 	if (stat)
58077497c49SRudolf Cornelissen 		LOG(4,("Brooktree: I2C errors occurred while setting mode PAL720\n"));
58177497c49SRudolf Cornelissen 
58277497c49SRudolf Cornelissen 	return stat;
58377497c49SRudolf Cornelissen }//end BT_init_PAL720.
58477497c49SRudolf Cornelissen 
58577497c49SRudolf Cornelissen static uint8 BT_init_NTSC720()
586fb6cadf1Sshatty {
58777497c49SRudolf Cornelissen 	uint8 stat;
588fb6cadf1Sshatty 
58977497c49SRudolf Cornelissen 	uint8 buffer[35];
59077497c49SRudolf Cornelissen 
5919a7218c5SRudolf Cornelissen 	LOG(4,("Brooktree: Setting NTSC 720x480 overscanning DVD mode\n"));
5929a7218c5SRudolf Cornelissen 
5939a7218c5SRudolf Cornelissen 	buffer[0] = si->ps.tv_encoder.adress + WR;	//issue I2C write command
59477497c49SRudolf Cornelissen 	buffer[1] = 0x76;		//select first bt register to write to.
59577497c49SRudolf Cornelissen 	buffer[2] = 0xf0;		//lsb h_clk_o: overscan comp = 0, so h_clk_o = 2 * h_clk_i (VSR=2 = scaling=1)
59677497c49SRudolf Cornelissen 	buffer[3] = 0xd0;		//lsb h_active: h_active = 720 pixels wide port
59777497c49SRudolf Cornelissen 	buffer[4] = 0x83;	//scope: OK hsync_width: (hsync_width / h_clk_o) * 63.55556uS = 4.70uS for NTSC
59877497c49SRudolf Cornelissen 	buffer[5] = 0x98;	//scope: OK	hburst_begin: (hburst_begin / h_clk_o) * 63.55556uS = 5.3uS for NTSC
59977497c49SRudolf Cornelissen 	buffer[6] = 0x5e;	//scope: OK hburst_end: ((hburst_end + 128) / h_clk_o) * 63.55556uS = 7.94uS for NTSC
60077497c49SRudolf Cornelissen 
60177497c49SRudolf Cornelissen 	//How to find the correct values for h_blank_o and v_blank_o:
60277497c49SRudolf Cornelissen 	// 1. Calculate h_blank_o according to initial setting guideline mentioned below;
60377497c49SRudolf Cornelissen 	// 2. Set v_blank_o in the neighbourhood of $18, so that TV picture does not have ghosts on right side in it while
60477497c49SRudolf Cornelissen 	//    horizontal position is about OK;
60577497c49SRudolf Cornelissen 	// 3. Then tune h_blank_o for centered output on scope (look at front porch and back porch);
60677497c49SRudolf Cornelissen 	// 4. Now shift the TV output using Hsync_offset for centered output while looking at TV (in method 'SetBT_Hphase' above);
60777497c49SRudolf Cornelissen 	// 5. If no vertical shivering occurs when image is centered, you're done. Else:
60877497c49SRudolf Cornelissen 	// 6. Modify the RIVA (BeScreen) h_sync_start setting somewhat to get stable centered picture possible on TV AND!:
60977497c49SRudolf Cornelissen 	// 7. Make sure you update the Chrontel horizontal Phase setting also then!
61077497c49SRudolf Cornelissen 
61177497c49SRudolf Cornelissen 	buffer[7] = 0x28;	//scope: tuned. lsb h_blank_o: h_blank_o = horizontal viewport location on TV
61277497c49SRudolf Cornelissen 						//(guideline for initial setting: (h_blank_o / h_clk_0) * 63.55556uS = 9.5uS for NTSC)
61377497c49SRudolf Cornelissen 	buffer[8] = 0x18;	//try-out; scope: checked against other modes, looks OK.	v_blank_o: 1e active line ('pixel')
61477497c49SRudolf Cornelissen 
61577497c49SRudolf Cornelissen 	buffer[9] = 0xf2;		//v_active_o: = (active output lines + 2) / field (on TV)
61677497c49SRudolf Cornelissen 	buffer[10] = 0x26;		//lsn = msn h_clk_o;
61777497c49SRudolf Cornelissen 							//b4-5 = msbits h_active;
61877497c49SRudolf Cornelissen 							//b7 = b8 v_avtive_o.
61977497c49SRudolf Cornelissen 	buffer[11] = 0x00;		//h_fract is always 0.
62077497c49SRudolf Cornelissen 	buffer[12] = 0x78;		//lsb h_clk_i: h_clk_i is horizontal total = 888.
62177497c49SRudolf Cornelissen 	buffer[13] = 0x90; 	//try-out; lsb h_blank_i: #clks between start sync and new line 1st pixel; copy to VGA delta-sync!
62277497c49SRudolf Cornelissen 	buffer[14] = 0x03;		//b2-0 = msn h_clk_i;
62377497c49SRudolf Cornelissen 						//try-out: b3 = msn h_blank_i;
62477497c49SRudolf Cornelissen 							//b4 = vblankdly is always 0.
62577497c49SRudolf Cornelissen 	buffer[15] = 0x0d;		//lsb v_lines_i: v_lines_i = 525
62677497c49SRudolf Cornelissen 	buffer[16] = 0x1a; 	//try-out;	v_blank_i: #input lines between start sync and new line (pixel); copy to VGA delta-sync!
62777497c49SRudolf Cornelissen 						//Make sure though that this value for the BT is *even*, or image will shiver a *lot* horizontally on TV.
62877497c49SRudolf Cornelissen 	buffer[17] = 0xe0;		//lsb v_active_i: v_active_i = 480
62977497c49SRudolf Cornelissen 	buffer[18] = 0x36;		//b1-0 = msn v_lines_i;
63077497c49SRudolf Cornelissen 							//b3-2 = msn v_active_i;
63177497c49SRudolf Cornelissen 							//b5-4 = ylpf = 3;
63277497c49SRudolf Cornelissen 							//b7-6 = clpf = 0.
63377497c49SRudolf Cornelissen 	buffer[19] = 0x00;		//lsb v_scale: v_scale = off = $1000
63477497c49SRudolf Cornelissen 	buffer[20] = 0x50;		//b5-0 = msn v_scale;
63577497c49SRudolf Cornelissen 						//scope: tuned. b7-6 = msn h_blank_o.
63677497c49SRudolf Cornelissen 							//(((PLL_INT + (PLL_FRACT/65536)) / 6) * 13500000) = PIXEL_CLK = (hor.tot. * v_lines_i * 60Hz)
63777497c49SRudolf Cornelissen 	buffer[21] = 0x98;		//lsb PLL fract: PLL fract = 0x6e98
63877497c49SRudolf Cornelissen 	buffer[22] = 0x6e;		//msb PLL fract
63977497c49SRudolf Cornelissen 	buffer[23] = 0x8c;		//b5-0 = PLL int: PLL int = 0x0c;
64077497c49SRudolf Cornelissen 							//b6 = by_pll: by_pll = 0;
64177497c49SRudolf Cornelissen 							//b7 = EN_XCLK: chip-pin CLKI is pixel clock.
64277497c49SRudolf Cornelissen 	buffer[24] = 0x0a;		//b0 = ni_out is always 0;
64377497c49SRudolf Cornelissen 							//b1 = setup = 1 for NTSC;
64477497c49SRudolf Cornelissen 							//b2 = 625line = 0 for NTSC;
64577497c49SRudolf Cornelissen 							//b3 = vsync_dur = 1 for NTSC;
64677497c49SRudolf Cornelissen 							//b4 = dic_screset is always 0;
64777497c49SRudolf Cornelissen 							//b5 = pal_md = 0 for NTSC;
64877497c49SRudolf Cornelissen 							//b6 = eclip is always 0;
64977497c49SRudolf Cornelissen 							//b7 = reserved (en_scart) is always 0.
65077497c49SRudolf Cornelissen 	buffer[25] = 0xe5;		//sync_amp $e5 for NTSC
65177497c49SRudolf Cornelissen 	buffer[26] = 0x75;		//bst_amp $74-$76 for NTSC
65277497c49SRudolf Cornelissen 	buffer[27] = 0x78;		//mcr: r-y $77-$79 for NTSC
65377497c49SRudolf Cornelissen 	buffer[28] = 0x44;		//mcb: b-y $43-$44 for NTSC
65477497c49SRudolf Cornelissen 	buffer[29] = 0x85;		//my: y $85 for NTSC
65577497c49SRudolf Cornelissen 	buffer[30] = 0x3c;		//lsb msc: msc b31-0: NTSC formula: ((3579545 / pixelclk) * 2^32) = MSC
65677497c49SRudolf Cornelissen 	buffer[31] = 0x91;		//msc = $20c2913c
65777497c49SRudolf Cornelissen 	buffer[32] = 0xc2;
65877497c49SRudolf Cornelissen 	buffer[33] = 0x20;		//msb msc.
65977497c49SRudolf Cornelissen 	buffer[34] = 0x00;		//phase_off always $00
66077497c49SRudolf Cornelissen 
66177497c49SRudolf Cornelissen 	/* reset status */
66277497c49SRudolf Cornelissen 	i2c_flag_error (-1);
66377497c49SRudolf Cornelissen 
66477497c49SRudolf Cornelissen 	i2c_bstart(si->ps.tv_encoder.bus);
66577497c49SRudolf Cornelissen 	i2c_writebuffer(si->ps.tv_encoder.bus, buffer, sizeof(buffer));
66677497c49SRudolf Cornelissen 	i2c_bstop(si->ps.tv_encoder.bus);
66777497c49SRudolf Cornelissen 	/* log on errors */
66877497c49SRudolf Cornelissen 	stat = i2c_flag_error(0);
66977497c49SRudolf Cornelissen 	if (stat)
67077497c49SRudolf Cornelissen 		LOG(4,("Brooktree: I2C errors occurred while setting mode NTSC720\n"));
67177497c49SRudolf Cornelissen 
67277497c49SRudolf Cornelissen 	return stat;
67377497c49SRudolf Cornelissen }//end BT_init_NTSC720.
67477497c49SRudolf Cornelissen 
67577497c49SRudolf Cornelissen static uint8 BT_init_PAL800_OS()
676fb6cadf1Sshatty {
67777497c49SRudolf Cornelissen 	uint8 stat;
678fb6cadf1Sshatty 
67977497c49SRudolf Cornelissen 	uint8 buffer[35];
68077497c49SRudolf Cornelissen 
6819a7218c5SRudolf Cornelissen 	LOG(4,("Brooktree: Setting PAL 800x600 overscanning VCD mode\n"));
6829a7218c5SRudolf Cornelissen 
6839a7218c5SRudolf Cornelissen 	buffer[0] = si->ps.tv_encoder.adress + WR;	//issue I2C write command
68477497c49SRudolf Cornelissen 	buffer[1] = 0x76;		//select first bt register to write to.
68577497c49SRudolf Cornelissen 	buffer[2] = 0x60;		//lsb h_clk_o: overscan comp = 0, so h_clk_o = 2 * h_clk_i (VSR=2 = scaling=1)
68677497c49SRudolf Cornelissen 	buffer[3] = 0x20;		//lsb h_active: h_active = 800 pixels wide port
68777497c49SRudolf Cornelissen 	buffer[4] = 0x8b;	//scope: OK hsync_width: (hsync_width / h_clk_o) * 64.0uS = 4.70uS for PAL
68877497c49SRudolf Cornelissen 	buffer[5] = 0xa5;	//scope: OK	hburst_begin: (hburst_begin / h_clk_o) * 64.0uS = 5.6uS for PAL
68977497c49SRudolf Cornelissen 	buffer[6] = 0x6b;	//scope: OK hburst_end: ((hburst_end + 128) / h_clk_o) * 64.0uS = 7.97uS for PAL
69077497c49SRudolf Cornelissen 
69177497c49SRudolf Cornelissen 	//How to find the correct values for h_blank_o and v_blank_o:
69277497c49SRudolf Cornelissen 	// 1. Calculate h_blank_o according to initial setting guideline mentioned below;
69377497c49SRudolf Cornelissen 	// 2. Set v_blank_o in the neighbourhood of $18, so that TV picture does not have ghosts on right side in it while
69477497c49SRudolf Cornelissen 	//    horizontal position is about OK;
69577497c49SRudolf Cornelissen 	// 3. Then tune h_blank_o for centered output on scope (look at front porch and back porch);
69677497c49SRudolf Cornelissen 	// 4. Now shift the TV output using Hsync_offset for centered output while looking at TV (in method 'SetBT_Hphase' above);
69777497c49SRudolf Cornelissen 	// 5. If no vertical shivering occurs when image is centered, you're done. Else:
69877497c49SRudolf Cornelissen 	// 6. Modify the RIVA (BeScreen) h_sync_start setting somewhat to get stable centered picture possible on TV AND!:
69977497c49SRudolf Cornelissen 	// 7. Make sure you update the Chrontel horizontal Phase setting also then!
70077497c49SRudolf Cornelissen 
70177497c49SRudolf Cornelissen 	if (si->ps.tv_encoder.type >= CX25870)//set CX value
702fb6cadf1Sshatty 	{
70377497c49SRudolf Cornelissen 		buffer[7] = 0xf4;
70477497c49SRudolf Cornelissen 		buffer[8] = 0x17;
705fb6cadf1Sshatty 	}
706fb6cadf1Sshatty 	else
70777497c49SRudolf Cornelissen 	{	//set BT value
70877497c49SRudolf Cornelissen 		buffer[7] = 0xc0;//scope: tuned. lsb h_blank_o: h_blank_o = horizontal viewport location on TV
70977497c49SRudolf Cornelissen 						//(guideline for initial setting: (h_blank_o / h_clk_0) * 64.0uS = 10.0uS for PAL)
71077497c49SRudolf Cornelissen 		buffer[8] = 0x18;//try-out; scope: checked against other modes, looks OK.	v_blank_o: 1e active line ('pixel')
71177497c49SRudolf Cornelissen 	}
71277497c49SRudolf Cornelissen 
71377497c49SRudolf Cornelissen 	buffer[9] = 0x2e;		//v_active_o: = (active output lines + 2) / field (on TV)
71477497c49SRudolf Cornelissen 	buffer[10] = 0xb7;		//lsn = msn h_clk_o;
71577497c49SRudolf Cornelissen 							//b4-5 = msbits h_active;
71677497c49SRudolf Cornelissen 							//b7 = b8 v_avtive_o.
71777497c49SRudolf Cornelissen 	buffer[11] = 0x00;		//h_fract is always 0.
71877497c49SRudolf Cornelissen 	buffer[12] = 0xb0;		//lsb h_clk_i: h_clk_i is horizontal total = 944.
71977497c49SRudolf Cornelissen 
72077497c49SRudolf Cornelissen 	if (si->ps.tv_encoder.type >= CX25870)//set CX value
72177497c49SRudolf Cornelissen 		buffer[13] = 0x20;
72277497c49SRudolf Cornelissen 	else				//set BT value
72377497c49SRudolf Cornelissen 		buffer[13] = 0x14;//try-out; lsb h_blank_i: #clks between start sync and new line 1st pixel; copy to VGA delta-sync!
72477497c49SRudolf Cornelissen 
72577497c49SRudolf Cornelissen 	buffer[14] = 0x03;		//b2-0 = msn h_clk_i;
72677497c49SRudolf Cornelissen 					 	//try-out: b3 = msn h_blank_i;
72777497c49SRudolf Cornelissen 							//b4 = vblankdly is always 0.
72877497c49SRudolf Cornelissen 	buffer[15] = 0x71;		//lsb v_lines_i: v_lines_i = 625
72977497c49SRudolf Cornelissen 
73077497c49SRudolf Cornelissen 	if (si->ps.tv_encoder.type >= CX25870)//set CX value
73177497c49SRudolf Cornelissen 		buffer[16] = 0x08;
73277497c49SRudolf Cornelissen 	else				//set BT value
73377497c49SRudolf Cornelissen 		buffer[16] = 0x2a;//try-out; v_blank_i: #input lines between start sync and new line (pixel); copy to VGA delta-sync!
73477497c49SRudolf Cornelissen 					 	//Make sure though that this value for the BT is *even*, or image will shiver a *lot* horizontally on TV.
73577497c49SRudolf Cornelissen 
73677497c49SRudolf Cornelissen 	buffer[17] = 0x58;		//lsb v_active_i: v_active_i = 600
73777497c49SRudolf Cornelissen 	buffer[18] = 0x3a;		//b1-0 = msn v_lines_i;
73877497c49SRudolf Cornelissen 							//b3-2 = msn v_active_i;
73977497c49SRudolf Cornelissen 							//b5-4 = ylpf = 3;
74077497c49SRudolf Cornelissen 							//b7-6 = clpf = 0.
74177497c49SRudolf Cornelissen 	buffer[19] = 0x00;		//lsb v_scale: v_scale = off = $1000
74277497c49SRudolf Cornelissen 	buffer[20] = 0x10;		//b5-0 = msn v_scale;
74377497c49SRudolf Cornelissen 						//scope: tuned. b7-6 = msn h_blank_o.
74477497c49SRudolf Cornelissen 							//(((PLL_INT + (PLL_FRACT/65536)) / 6) * 13500000) = PIXEL_CLK = (hor.tot. * v_lines_i * 50Hz)
74577497c49SRudolf Cornelissen 	buffer[21] = 0x72;		//lsb PLL fract: PLL fract = 0x1c72
74677497c49SRudolf Cornelissen 	buffer[22] = 0x1c;		//msb PLL fract
74777497c49SRudolf Cornelissen 	buffer[23] = 0x8d;		//b5-0 = PLL int: PLL int = 0x0d;
74877497c49SRudolf Cornelissen 							//b6 = by_pll: by_pll = 0;
74977497c49SRudolf Cornelissen 							//b7 = EN_XCLK: chip-pin CLKI is pixel clock.
75077497c49SRudolf Cornelissen 	buffer[24] = 0x24;		//b0 = ni_out is always 0;
75177497c49SRudolf Cornelissen 							//b1 = setup = 0 for PAL;
75277497c49SRudolf Cornelissen 							//b2 = 625line = 1 for PAL;
75377497c49SRudolf Cornelissen 							//b3 = vsync_dur = 0 for PAL;
75477497c49SRudolf Cornelissen 							//b4 = dic_screset is always 0;
75577497c49SRudolf Cornelissen 							//b5 = pal_md = 1 for PAL;
75677497c49SRudolf Cornelissen 							//b6 = eclip is always 0;
75777497c49SRudolf Cornelissen 							//b7 = reserved (en_scart) is always 0.
75877497c49SRudolf Cornelissen 	buffer[25] = 0xf0;		//sync_amp $f0 for PAL
75977497c49SRudolf Cornelissen 	buffer[26] = 0x57;		//bst_amp $57-$58 for PAL
76077497c49SRudolf Cornelissen 	buffer[27] = 0x80;		//mcr: r-y $80-$81 for PAL
76177497c49SRudolf Cornelissen 	buffer[28] = 0x48;		//mcb: b-y $48-$49 for PAL
76277497c49SRudolf Cornelissen 	buffer[29] = 0x8c;		//my: y $8c for PAL
76377497c49SRudolf Cornelissen 	buffer[30] = 0x31;		//lsb msc: msc b31-0: PAL formula: ((4433619 / pixelclk) * 2^32) = MSC
76477497c49SRudolf Cornelissen 	buffer[31] = 0x8c;		//msc = $26798c31
76577497c49SRudolf Cornelissen 	buffer[32] = 0x79;
76677497c49SRudolf Cornelissen 	buffer[33] = 0x26;		//msb msc.
76777497c49SRudolf Cornelissen 	buffer[34] = 0x00;		//phase_off always $00
76877497c49SRudolf Cornelissen 
76977497c49SRudolf Cornelissen 	/* reset status */
77077497c49SRudolf Cornelissen 	i2c_flag_error (-1);
77177497c49SRudolf Cornelissen 
77277497c49SRudolf Cornelissen 	i2c_bstart(si->ps.tv_encoder.bus);
77377497c49SRudolf Cornelissen 	i2c_writebuffer(si->ps.tv_encoder.bus, buffer, sizeof(buffer));
77477497c49SRudolf Cornelissen 	i2c_bstop(si->ps.tv_encoder.bus);
77577497c49SRudolf Cornelissen 	/* log on errors */
77677497c49SRudolf Cornelissen 	stat = i2c_flag_error(0);
77777497c49SRudolf Cornelissen 	if (stat)
77877497c49SRudolf Cornelissen 		LOG(4,("Brooktree: I2C errors occurred while setting mode PAL800 OS\n"));
77977497c49SRudolf Cornelissen 
78077497c49SRudolf Cornelissen 	return stat;
78177497c49SRudolf Cornelissen }//end BT_init_PAL800_OS.
78277497c49SRudolf Cornelissen 
78377497c49SRudolf Cornelissen static uint8 BT_init_NTSC640_OS()
784fb6cadf1Sshatty {
78577497c49SRudolf Cornelissen 	uint8 stat;
786fb6cadf1Sshatty 
78777497c49SRudolf Cornelissen 	uint8 buffer[35];
788fb6cadf1Sshatty 
7899a7218c5SRudolf Cornelissen 	LOG(4,("Brooktree: Setting NTSC 640x480 overscanning VCD mode\n"));
7909a7218c5SRudolf Cornelissen 
7919a7218c5SRudolf Cornelissen 	buffer[0] = si->ps.tv_encoder.adress + WR;	//issue I2C write command
79277497c49SRudolf Cornelissen 	buffer[1] = 0x76;		//select first bt register to write to.
79377497c49SRudolf Cornelissen 	buffer[2] = 0x20;		//lsb h_clk_o: overscan comp = 0, so h_clk_o = 2 * h_clk_i (VSR=2 = scaling=1)
79477497c49SRudolf Cornelissen 	buffer[3] = 0x80;		//lsb h_active: h_active = 640 pixels wide port
79577497c49SRudolf Cornelissen 	buffer[4] = 0x74;	//scope: OK hsync_width: (hsync_width / h_clk_o) * 63.55556uS = 4.70uS for NTSC
79677497c49SRudolf Cornelissen 	buffer[5] = 0x83;	//scope: OK	hburst_begin: (hburst_begin / h_clk_o) * 63.55556uS = 5.3uS for NTSC
79777497c49SRudolf Cornelissen 	buffer[6] = 0x44;	//scope: OK hburst_end: ((hburst_end + 128) / h_clk_o) * 63.55556uS = 7.94uS for NTSC
798fb6cadf1Sshatty 
79977497c49SRudolf Cornelissen 	//How to find the correct values for h_blank_o and v_blank_o:
80077497c49SRudolf Cornelissen 	// 1. Calculate h_blank_o according to initial setting guideline mentioned below;
80177497c49SRudolf Cornelissen 	// 2. Set v_blank_o in the neighbourhood of $18, so that TV picture does not have ghosts on right side in it while
80277497c49SRudolf Cornelissen 	//    horizontal position is about OK;
80377497c49SRudolf Cornelissen 	// 3. Then tune h_blank_o for centered output on scope (look at front porch and back porch);
80477497c49SRudolf Cornelissen 	// 4. Now shift the TV output using Hsync_offset for centered output while looking at TV (in method 'SetBT_Hphase' above);
80577497c49SRudolf Cornelissen 	// 5. If no vertical shivering occurs when image is centered, you're done. Else:
80677497c49SRudolf Cornelissen 	// 6. Modify the RIVA (BeScreen) h_sync_start setting somewhat to get stable centered picture possible on TV AND!:
80777497c49SRudolf Cornelissen 	// 7. Make sure you update the Chrontel horizontal Phase setting also then!
808fb6cadf1Sshatty 
80977497c49SRudolf Cornelissen 	buffer[7] = 0xf7;	//scope: tuned. lsb h_blank_o: h_blank_o = horizontal viewport location on TV:
81077497c49SRudolf Cornelissen 						//(guideline for initial setting: (h_blank_o / h_clk_0) * 63.55556uS = 9.5uS for NTSC)
811fb6cadf1Sshatty 
81277497c49SRudolf Cornelissen 	if (si->ps.tv_encoder.type >= CX25870)//set CX value
81377497c49SRudolf Cornelissen 		buffer[8] = 0x1d;
81477497c49SRudolf Cornelissen 	else				//set BT value
81577497c49SRudolf Cornelissen 		buffer[8] = 0x1c;//try-out; scope: checked against other modes, looks OK.	v_blank_o: 1e active line ('pixel')
816fb6cadf1Sshatty 
81777497c49SRudolf Cornelissen 	buffer[9] = 0xf2;		//v_active_o: = (active output lines + 2) / field (on TV)
81877497c49SRudolf Cornelissen 	buffer[10] = 0x26;		//lsn = msn h_clk_o;
81977497c49SRudolf Cornelissen 							//b4-5 = msbits h_active;
82077497c49SRudolf Cornelissen 							//b7 = b8 v_avtive_o.
82177497c49SRudolf Cornelissen 	buffer[11] = 0x00;		//h_fract is always 0.
82277497c49SRudolf Cornelissen 	buffer[12] = 0x10;		//lsb h_clk_i: h_clk_i is horizontal total = 784.
82377497c49SRudolf Cornelissen 	buffer[13] = 0x14;	//try-out; lsb h_blank_i: #clks between start sync and new line 1st pixel; copy to VGA delta-sync!
82477497c49SRudolf Cornelissen 	buffer[14] = 0x03;		//b2-0 = msn h_clk_i;
82577497c49SRudolf Cornelissen 						//try-out: b3 = msn h_blank_i;
82677497c49SRudolf Cornelissen 							//b4 = vblankdly is always 0.
82777497c49SRudolf Cornelissen 	buffer[15] = 0x0d;		//lsb v_lines_i: v_lines_i = 525
82877497c49SRudolf Cornelissen 	buffer[16] = 0x18;	//try-out;	v_blank_i: #input lines between start sync and new line (pixel); copy to VGA delta-sync!
82977497c49SRudolf Cornelissen 					 	//Make sure though that this value for the BT is *even*, or image will shiver a *lot* horizontally on TV.
83077497c49SRudolf Cornelissen 	buffer[17] = 0xe0;		//lsb v_active_i: v_active_i = 480
83177497c49SRudolf Cornelissen 	buffer[18] = 0x36;		//b1-0 = msn v_lines_i;
83277497c49SRudolf Cornelissen 							//b3-2 = msn v_active_i;
83377497c49SRudolf Cornelissen 							//b5-4 = ylpf = 3;
83477497c49SRudolf Cornelissen 							//b7-6 = clpf = 0.
83577497c49SRudolf Cornelissen 	buffer[19] = 0x00;		//lsb v_scale: v_scale = off = $1000
83677497c49SRudolf Cornelissen 	buffer[20] = 0x10;		//b5-0 = msn v_scale;
83777497c49SRudolf Cornelissen 						//scope: tuned. b7-6 = msn h_blank_o.
83877497c49SRudolf Cornelissen 							//(((PLL_INT + (PLL_FRACT/65536)) / 6) * 13500000) = PIXEL_CLK = (hor.tot. * v_lines_i * 60Hz)
83977497c49SRudolf Cornelissen 	buffer[21] = 0xdb;		//lsb PLL fract: PLL fract = 0xf9db
84077497c49SRudolf Cornelissen 	buffer[22] = 0xf9;		//msb PLL fract
84177497c49SRudolf Cornelissen 	buffer[23] = 0x8a;		//b5-0 = PLL int: PLL int = 0x0a;
84277497c49SRudolf Cornelissen 							//b6 = by_pll: by_pll = 0;
84377497c49SRudolf Cornelissen 							//b7 = EN_XCLK: chip-pin CLKI is pixel clock.
84477497c49SRudolf Cornelissen 	buffer[24] = 0x0a;		//b0 = ni_out is always 0;
84577497c49SRudolf Cornelissen 							//b1 = setup = 1 for NTSC;
84677497c49SRudolf Cornelissen 							//b2 = 625line = 0 for NTSC;
84777497c49SRudolf Cornelissen 							//b3 = vsync_dur = 1 for NTSC;
84877497c49SRudolf Cornelissen 							//b4 = dic_screset is always 0;
84977497c49SRudolf Cornelissen 							//b5 = pal_md = 0 for NTSC;
85077497c49SRudolf Cornelissen 							//b6 = eclip is always 0;
85177497c49SRudolf Cornelissen 							//b7 = reserved (en_scart) is always 0.
85277497c49SRudolf Cornelissen 	buffer[25] = 0xe5;		//sync_amp $e5 for NTSC
85377497c49SRudolf Cornelissen 	buffer[26] = 0x75;		//bst_amp $74-$76 for NTSC
85477497c49SRudolf Cornelissen 	buffer[27] = 0x78;		//mcr: r-y $77-$79 for NTSC
85577497c49SRudolf Cornelissen 	buffer[28] = 0x44;		//mcb: b-y $43-$44 for NTSC
85677497c49SRudolf Cornelissen 	buffer[29] = 0x85;		//my: y $85 for NTSC
85777497c49SRudolf Cornelissen 	buffer[30] = 0x37;		//lsb msc: msc b31-0: NTSC formula: ((3579545 / pixelclk) * 2^32) = MSC
85877497c49SRudolf Cornelissen 	buffer[31] = 0x12;		//msc = $251b1237
85977497c49SRudolf Cornelissen 	buffer[32] = 0x1b;
86077497c49SRudolf Cornelissen 	buffer[33] = 0x25;		//msb msc.
86177497c49SRudolf Cornelissen 	buffer[34] = 0x00;		//phase_off always $00
862fb6cadf1Sshatty 
86377497c49SRudolf Cornelissen 	/* reset status */
86477497c49SRudolf Cornelissen 	i2c_flag_error (-1);
865fb6cadf1Sshatty 
86677497c49SRudolf Cornelissen 	i2c_bstart(si->ps.tv_encoder.bus);
86777497c49SRudolf Cornelissen 	i2c_writebuffer(si->ps.tv_encoder.bus, buffer, sizeof(buffer));
86877497c49SRudolf Cornelissen 	i2c_bstop(si->ps.tv_encoder.bus);
86977497c49SRudolf Cornelissen 	/* log on errors */
87077497c49SRudolf Cornelissen 	stat = i2c_flag_error(0);
87177497c49SRudolf Cornelissen 	if (stat)
87277497c49SRudolf Cornelissen 		LOG(4,("Brooktree: I2C errors occurred while setting mode NTSC640 OS\n"));
873fb6cadf1Sshatty 
87477497c49SRudolf Cornelissen 	return stat;
87577497c49SRudolf Cornelissen }//end BT_init_NTSC640_OS.
876fb6cadf1Sshatty 
87721f6ecabSRudolf Cornelissen static uint8 BT_testsignal(void)
87821f6ecabSRudolf Cornelissen {
87921f6ecabSRudolf Cornelissen 	uint8 stat;
88021f6ecabSRudolf Cornelissen 
88121f6ecabSRudolf Cornelissen 	uint8 buffer[3];
88221f6ecabSRudolf Cornelissen 
8839a7218c5SRudolf Cornelissen 	LOG(4,("Brooktree: Enabling testsignal\n"));
8849a7218c5SRudolf Cornelissen 
88521f6ecabSRudolf Cornelissen 	buffer[0] = si->ps.tv_encoder.adress + WR;
88621f6ecabSRudolf Cornelissen 	/* select bt register for enabling colorbars and outputs */
88721f6ecabSRudolf Cornelissen 	buffer[1] = 0xc4;
88821f6ecabSRudolf Cornelissen 	/* issue the actual command */
88921f6ecabSRudolf Cornelissen 	buffer[2] = 0x05;
89021f6ecabSRudolf Cornelissen 
89121f6ecabSRudolf Cornelissen 	/* reset status */
89221f6ecabSRudolf Cornelissen 	i2c_flag_error (-1);
89321f6ecabSRudolf Cornelissen 
89421f6ecabSRudolf Cornelissen 	i2c_bstart(si->ps.tv_encoder.bus);
89521f6ecabSRudolf Cornelissen 	i2c_writebuffer(si->ps.tv_encoder.bus, buffer, sizeof(buffer));
89621f6ecabSRudolf Cornelissen 	i2c_bstop(si->ps.tv_encoder.bus);
89721f6ecabSRudolf Cornelissen 	/* log on errors */
89821f6ecabSRudolf Cornelissen 	stat = i2c_flag_error(0);
89921f6ecabSRudolf Cornelissen 	if (stat)
90021f6ecabSRudolf Cornelissen 		LOG(4,("Brooktree: I2C errors occurred while setting up flickerfilter and outputs\n"));
90121f6ecabSRudolf Cornelissen 
90221f6ecabSRudolf Cornelissen 	return stat;
90321f6ecabSRudolf Cornelissen }//end BT_testsignal.
90421f6ecabSRudolf Cornelissen 
90521f6ecabSRudolf Cornelissen static uint8 BT_setup_output(uint8 monstat, uint8 output, uint8 ffilter)
90621f6ecabSRudolf Cornelissen {
90721f6ecabSRudolf Cornelissen 	uint8 stat;
90821f6ecabSRudolf Cornelissen 
90921f6ecabSRudolf Cornelissen 	uint8 buffer[7];
91021f6ecabSRudolf Cornelissen 
91121f6ecabSRudolf Cornelissen 	buffer[0] = si->ps.tv_encoder.adress + WR;
91221f6ecabSRudolf Cornelissen 	/* select first TV config register to write */
91321f6ecabSRudolf Cornelissen 	buffer[1] = 0xc6;
91421f6ecabSRudolf Cornelissen 	/* input is 24bit mpx'd RGB, BLANK = out, sync = act. hi */
91521f6ecabSRudolf Cornelissen 	buffer[2] = 0x98;
91621f6ecabSRudolf Cornelissen 	/* disable all filters, exept flicker filter */
91721f6ecabSRudolf Cornelissen 	buffer[3] = 0x98;
91821f6ecabSRudolf Cornelissen 	if (!ffilter)
91921f6ecabSRudolf Cornelissen 	{
92021f6ecabSRudolf Cornelissen 		/* disable flicker filter */
92121f6ecabSRudolf Cornelissen 		buffer[3] = 0xc0;
9229a7218c5SRudolf Cornelissen 		LOG(4,("Brooktree: Disabling flickerfilter\n"));
92321f6ecabSRudolf Cornelissen 	}
9249a7218c5SRudolf Cornelissen 	else
9259a7218c5SRudolf Cornelissen 		LOG(4,("Brooktree: Enabling flickerfilter\n"));
9269a7218c5SRudolf Cornelissen 
92721f6ecabSRudolf Cornelissen 	/* (disable filters) */
92821f6ecabSRudolf Cornelissen 	buffer[4] = 0xc0;
92921f6ecabSRudolf Cornelissen 	/* (disable filters) */
93021f6ecabSRudolf Cornelissen 	buffer[5] = 0xc0;
93121f6ecabSRudolf Cornelissen 	switch (output)
93221f6ecabSRudolf Cornelissen 	/* Description of ELSA Erazor III hardware layout:
93321f6ecabSRudolf Cornelissen 	 * (This is the default (recommended) layout by NVIDIA)
93421f6ecabSRudolf Cornelissen 	 * DAC A = CVBS
93521f6ecabSRudolf Cornelissen 	 * DAC B = C (chrominance)
93621f6ecabSRudolf Cornelissen 	 * DAC C = Y (luminance) */
93721f6ecabSRudolf Cornelissen 
93821f6ecabSRudolf Cornelissen 	/* Description of Diamond VIPER550:
93921f6ecabSRudolf Cornelissen 	 * DAC A = Not connected
94021f6ecabSRudolf Cornelissen 	 * DAC B = C (chrominance)
94121f6ecabSRudolf Cornelissen 	 * DAC C = Y (luminance)
94221f6ecabSRudolf Cornelissen 	 * To be able to connect to CVBS TV's a special cable is supplied:
94321f6ecabSRudolf Cornelissen 	 * This cable connects the Y (DAC C) output to the TV CVBS input. */
94421f6ecabSRudolf Cornelissen 	{
9459a7218c5SRudolf Cornelissen 	case 1:
9469a7218c5SRudolf Cornelissen 		LOG(4,("Brooktree: Outputting both Y/C and CVBS where supported by hardware\n"));
9479a7218c5SRudolf Cornelissen 		buffer[6] = 0x18;	// Y/C and CVBS out if all ports implemented
94821f6ecabSRudolf Cornelissen 							// in hardware, else only Y/C or CVBS out.
94921f6ecabSRudolf Cornelissen 		break;
9509a7218c5SRudolf Cornelissen 	case 2:
9519a7218c5SRudolf Cornelissen 		LOG(4,("Brooktree: Outputting CVBS on all outputs\n"));
9529a7218c5SRudolf Cornelissen 		buffer[6] = 0x00;	// put CVBS on all outputs. Used for cards
95321f6ecabSRudolf Cornelissen 		break;				// with only Y/C out and 'translation cable'.
9549a7218c5SRudolf Cornelissen 	default:
9559a7218c5SRudolf Cornelissen 		LOG(4,("Brooktree: Outputting signals according to autodetect status\n"));
9569a7218c5SRudolf Cornelissen 		switch (monstat)	// only 'autodetect' remains...
95721f6ecabSRudolf Cornelissen 		{
9589a7218c5SRudolf Cornelissen 		case 1:
9599a7218c5SRudolf Cornelissen 			buffer[6] = 0x00;	//only Y connected: must be CVBS!
96021f6ecabSRudolf Cornelissen 			break;
9619a7218c5SRudolf Cornelissen 		case 2:
9629a7218c5SRudolf Cornelissen 			buffer[6] = 0x00;	//only C connected: must be CVBS!
96321f6ecabSRudolf Cornelissen 			break;				//(though cable is wired wrong...)
9649a7218c5SRudolf Cornelissen 		case 5:
9659a7218c5SRudolf Cornelissen 			buffer[6] = 0x00;	//CVBS and only Y connected: 2x CVBS!
96621f6ecabSRudolf Cornelissen 			break;			   	//(officially not supported...)
9679a7218c5SRudolf Cornelissen 		case 6:
9689a7218c5SRudolf Cornelissen 			buffer[6] = 0x00;	//CVBS and only C connected: 2x CVBS!
96921f6ecabSRudolf Cornelissen 			break;			   	//(officially not supported...)
9709a7218c5SRudolf Cornelissen 		default:
9719a7218c5SRudolf Cornelissen 			buffer[6] = 0x18;	//nothing, or
97221f6ecabSRudolf Cornelissen 							 	//Y/C only, or
97321f6ecabSRudolf Cornelissen 							 	//CVBS only (but on CVBS output), or
97421f6ecabSRudolf Cornelissen 							 	//Y/C and CVBS connected:
97521f6ecabSRudolf Cornelissen 							 	//So activate recommended signals.
97621f6ecabSRudolf Cornelissen 		}
97721f6ecabSRudolf Cornelissen 	}
97821f6ecabSRudolf Cornelissen 
97921f6ecabSRudolf Cornelissen 	/* reset status */
98021f6ecabSRudolf Cornelissen 	i2c_flag_error (-1);
98121f6ecabSRudolf Cornelissen 
98221f6ecabSRudolf Cornelissen 	i2c_bstart(si->ps.tv_encoder.bus);
98321f6ecabSRudolf Cornelissen 	i2c_writebuffer(si->ps.tv_encoder.bus, buffer, sizeof(buffer));
98421f6ecabSRudolf Cornelissen 	i2c_bstop(si->ps.tv_encoder.bus);
98521f6ecabSRudolf Cornelissen 	/* log on errors */
98621f6ecabSRudolf Cornelissen 	stat = i2c_flag_error(0);
98721f6ecabSRudolf Cornelissen 	if (stat)
98821f6ecabSRudolf Cornelissen 		LOG(4,("Brooktree: I2C errors occurred while setting up flickerfilter and outputs\n"));
98921f6ecabSRudolf Cornelissen 
99021f6ecabSRudolf Cornelissen 	return stat;
99121f6ecabSRudolf Cornelissen }//end BT_setup_output.
99221f6ecabSRudolf Cornelissen 
99321f6ecabSRudolf Cornelissen static uint8 BT_setup_hphase(uint8 mode)
99421f6ecabSRudolf Cornelissen {
99521f6ecabSRudolf Cornelissen 	uint8 stat, hoffset;
99621f6ecabSRudolf Cornelissen 
99721f6ecabSRudolf Cornelissen 	uint8 buffer[7];
99821f6ecabSRudolf Cornelissen 
9999a7218c5SRudolf Cornelissen 	LOG(4,("Brooktree: Tuning horizontal phase\n"));
10009a7218c5SRudolf Cornelissen 
100121f6ecabSRudolf Cornelissen 	/* CX needs timing reset (advised on BT also), first 1mS delay needed! */
100221f6ecabSRudolf Cornelissen 	snooze(1000);
100321f6ecabSRudolf Cornelissen 
100421f6ecabSRudolf Cornelissen 	/* values below are all tested on TNT1, TNT2 and GeForce2MX */
100521f6ecabSRudolf Cornelissen 	buffer[0] = si->ps.tv_encoder.adress + WR;
100621f6ecabSRudolf Cornelissen 	/* select first TV output timing register to write */
100721f6ecabSRudolf Cornelissen 	buffer[1] = 0x6c;
100821f6ecabSRudolf Cornelissen 	/* turn on active video & generate timing reset on CX chips! */
100921f6ecabSRudolf Cornelissen 	buffer[2] = 0x86;
101021f6ecabSRudolf Cornelissen 	/* (set fail save values...) */
101121f6ecabSRudolf Cornelissen 	buffer[3] = 0x00;		//set default horizontal sync offset
101221f6ecabSRudolf Cornelissen 	buffer[4] = 0x02;		//set default horizontal sync width
101321f6ecabSRudolf Cornelissen 	buffer[5] = 0x00;		//set default vertical sync offset
101421f6ecabSRudolf Cornelissen 
101521f6ecabSRudolf Cornelissen 	/* do specific timing setup for all chips and modes: */
101621f6ecabSRudolf Cornelissen 	hoffset = 8;					//TNT2 and TNT2 M64...(8 pixels delayed hpos,
101721f6ecabSRudolf Cornelissen 									//so picture more to the right)
101821f6ecabSRudolf Cornelissen //fixme: check this out...
101921f6ecabSRudolf Cornelissen //	if (ddata->type == 1) hoffset = 0;	//TNT1 or GeForce2... (std hpos)
102021f6ecabSRudolf Cornelissen 									//NOTE: It might be that GeForce needs TNT2 offset:
102121f6ecabSRudolf Cornelissen 									//for now CX chips get seperate extra offset, until sure.
102221f6ecabSRudolf Cornelissen 									//(CX is only found AFAIK on GeForce cards, no BT tested
102321f6ecabSRudolf Cornelissen 									// on GeForce yet. CH was tested on GeForce and seemed to
102421f6ecabSRudolf Cornelissen 									// indicate TNT1 offset was needed.)
102521f6ecabSRudolf Cornelissen 
102621f6ecabSRudolf Cornelissen 	switch (mode)
102721f6ecabSRudolf Cornelissen 	{
102821f6ecabSRudolf Cornelissen 	case NTSC640_TST:
102921f6ecabSRudolf Cornelissen 	case NTSC640:
103021f6ecabSRudolf Cornelissen 		if (si->ps.tv_encoder.type >= CX25870) hoffset +=8; //if CX shift picture right some more...
103121f6ecabSRudolf Cornelissen 		buffer[3] = (0x25 + hoffset);	//set horizontal sync offset
103221f6ecabSRudolf Cornelissen 		break;
103321f6ecabSRudolf Cornelissen 	case NTSC800:
103421f6ecabSRudolf Cornelissen 		if (si->ps.tv_encoder.type >= CX25870) hoffset +=8; //if CX shift picture right some more...
103521f6ecabSRudolf Cornelissen 		buffer[3] = (0xe1 + hoffset);	//set horizontal sync offset
103621f6ecabSRudolf Cornelissen 		buffer[4] = 0xc2;
103721f6ecabSRudolf Cornelissen 		//Vsync offset reg. does not exist on CX: mode is checked and OK.
103821f6ecabSRudolf Cornelissen 		buffer[5] = 0x40;				//set VSync offset (on BT's only)
103921f6ecabSRudolf Cornelissen 		break;
104021f6ecabSRudolf Cornelissen 	case PAL640:
104121f6ecabSRudolf Cornelissen 		if (si->ps.tv_encoder.type >= CX25870) hoffset +=8; //if CX shift picture right some more...
104221f6ecabSRudolf Cornelissen 		buffer[3] = (0xa8 + hoffset);
104321f6ecabSRudolf Cornelissen 		break;
104421f6ecabSRudolf Cornelissen 	case PAL800_TST:
104521f6ecabSRudolf Cornelissen 	case PAL800:
104621f6ecabSRudolf Cornelissen 		if (si->ps.tv_encoder.type >= CX25870) hoffset +=8; //if CX shift picture right some more...
104721f6ecabSRudolf Cornelissen 		buffer[3] = (0x2c + hoffset);
104821f6ecabSRudolf Cornelissen 		break;
104921f6ecabSRudolf Cornelissen 	case NTSC720:
105021f6ecabSRudolf Cornelissen 		if (si->ps.tv_encoder.type >= CX25870)
105121f6ecabSRudolf Cornelissen 			buffer[3] = (0xb2 + hoffset); //set horizontal sync offset CX
105221f6ecabSRudolf Cornelissen 		else
105321f6ecabSRudolf Cornelissen 			buffer[3] = (0xd0 + hoffset); //set horizontal sync offset BT
105421f6ecabSRudolf Cornelissen 		buffer[4] = 0xff;				//hsync width = max:
105521f6ecabSRudolf Cornelissen 		break;							//to prevent vertical image 'shivering'.
105621f6ecabSRudolf Cornelissen 	case PAL720:
105721f6ecabSRudolf Cornelissen 		buffer[3] = (0xd4 + hoffset);
105821f6ecabSRudolf Cornelissen 		buffer[4] = 0xff;
105921f6ecabSRudolf Cornelissen 		break;
106021f6ecabSRudolf Cornelissen 	case NTSC640_OS:
106121f6ecabSRudolf Cornelissen 		buffer[3] = (0xc8 + hoffset);
106221f6ecabSRudolf Cornelissen 		buffer[4] = 0xff;
106321f6ecabSRudolf Cornelissen 		break;
106421f6ecabSRudolf Cornelissen 	case PAL800_OS:
106521f6ecabSRudolf Cornelissen 		if (si->ps.tv_encoder.type >= CX25870)
106621f6ecabSRudolf Cornelissen 			buffer[3] = (0x78 + hoffset); //set horizontal sync offset CX
106721f6ecabSRudolf Cornelissen 		else
106821f6ecabSRudolf Cornelissen 			buffer[3] = (0xc4 + hoffset); //set horizontal sync offset BT
106921f6ecabSRudolf Cornelissen 		buffer[4] = 0xff;
107021f6ecabSRudolf Cornelissen 		break;
107121f6ecabSRudolf Cornelissen 	default: //nothing to be done here...
107221f6ecabSRudolf Cornelissen 		break;
107321f6ecabSRudolf Cornelissen 	}
107421f6ecabSRudolf Cornelissen 
107521f6ecabSRudolf Cornelissen 	buffer[6] = 0x01;		//set default vertical sync width
107621f6ecabSRudolf Cornelissen 
107721f6ecabSRudolf Cornelissen 	/* reset status */
107821f6ecabSRudolf Cornelissen 	i2c_flag_error (-1);
107921f6ecabSRudolf Cornelissen 
108021f6ecabSRudolf Cornelissen 	i2c_bstart(si->ps.tv_encoder.bus);
108121f6ecabSRudolf Cornelissen 	i2c_writebuffer(si->ps.tv_encoder.bus, buffer, sizeof(buffer));
108221f6ecabSRudolf Cornelissen 	i2c_bstop(si->ps.tv_encoder.bus);
108321f6ecabSRudolf Cornelissen 	/* log on errors */
108421f6ecabSRudolf Cornelissen 	stat = i2c_flag_error(0);
108521f6ecabSRudolf Cornelissen 	if (stat)
108621f6ecabSRudolf Cornelissen 		LOG(4,("Brooktree: I2C errors occurred while setting up h_phase\n"));
108721f6ecabSRudolf Cornelissen 
108821f6ecabSRudolf Cornelissen 	return stat;
108921f6ecabSRudolf Cornelissen }//end BT_setup_hphase.
109021f6ecabSRudolf Cornelissen 
109121f6ecabSRudolf Cornelissen static uint8 BT_read_monstat(uint8* monstat)
109221f6ecabSRudolf Cornelissen {
109321f6ecabSRudolf Cornelissen 	uint8 stat;
109421f6ecabSRudolf Cornelissen 	uint8 buffer[3];
109521f6ecabSRudolf Cornelissen 
1096efe9c7a6SRudolf Cornelissen 	/* make sure we have the recommended failsafe selected */
1097efe9c7a6SRudolf Cornelissen 	*monstat = 0;
1098efe9c7a6SRudolf Cornelissen 
10999a7218c5SRudolf Cornelissen 	LOG(4,("Brooktree: Autodetecting connected output devices\n"));
11009a7218c5SRudolf Cornelissen 
110121f6ecabSRudolf Cornelissen 	/* set BT to return connection status in ESTATUS on next read CMD: */
110221f6ecabSRudolf Cornelissen 	buffer[0] = si->ps.tv_encoder.adress + WR;
110321f6ecabSRudolf Cornelissen 	/* set ESTATUS at b'01' (return conn.stat.) */
110421f6ecabSRudolf Cornelissen 	buffer[1] = 0xc4;
110521f6ecabSRudolf Cornelissen 	/* and leave chip outputs on. */
110621f6ecabSRudolf Cornelissen 	buffer[2] = 0x41;
110721f6ecabSRudolf Cornelissen 
110821f6ecabSRudolf Cornelissen 	/* reset status */
110921f6ecabSRudolf Cornelissen 	i2c_flag_error (-1);
111021f6ecabSRudolf Cornelissen 
111121f6ecabSRudolf Cornelissen 	i2c_bstart(si->ps.tv_encoder.bus);
111221f6ecabSRudolf Cornelissen 	i2c_writebuffer(si->ps.tv_encoder.bus, buffer, sizeof(buffer));
111321f6ecabSRudolf Cornelissen 	i2c_bstop(si->ps.tv_encoder.bus);
111421f6ecabSRudolf Cornelissen 	/* log on errors */
111521f6ecabSRudolf Cornelissen 	stat = i2c_flag_error(0);
111621f6ecabSRudolf Cornelissen 	if (stat)
111721f6ecabSRudolf Cornelissen 	{
111821f6ecabSRudolf Cornelissen 		LOG(4,("Brooktree: I2C errors occurred while reading connection status (1)\n"));
111921f6ecabSRudolf Cornelissen 		return stat;
112021f6ecabSRudolf Cornelissen 	}
112121f6ecabSRudolf Cornelissen 
112221f6ecabSRudolf Cornelissen 	/* do actual read connection status: */
112321f6ecabSRudolf Cornelissen 	buffer[0] = si->ps.tv_encoder.adress + WR;
112421f6ecabSRudolf Cornelissen 	/* select register with CHECK_STAT CMD */
112521f6ecabSRudolf Cornelissen 	buffer[1] = 0xba;
112621f6ecabSRudolf Cornelissen 	/* issue actual command. */
112721f6ecabSRudolf Cornelissen 	buffer[2] = 0x40;
112821f6ecabSRudolf Cornelissen 
112921f6ecabSRudolf Cornelissen 	i2c_bstart(si->ps.tv_encoder.bus);
113021f6ecabSRudolf Cornelissen 	i2c_writebuffer(si->ps.tv_encoder.bus, buffer, sizeof(buffer));
113121f6ecabSRudolf Cornelissen 	i2c_bstop(si->ps.tv_encoder.bus);
113221f6ecabSRudolf Cornelissen 	/* log on errors */
113321f6ecabSRudolf Cornelissen 	stat = i2c_flag_error(0);
113421f6ecabSRudolf Cornelissen 	if (stat)
113521f6ecabSRudolf Cornelissen 	{
113621f6ecabSRudolf Cornelissen 		LOG(4,("Brooktree: I2C errors occurred while reading connection status (2)\n"));
113721f6ecabSRudolf Cornelissen 		return stat;
113821f6ecabSRudolf Cornelissen 	}
113921f6ecabSRudolf Cornelissen 
114021f6ecabSRudolf Cornelissen 	/* CX: Wait 600uS for signals to stabilize (see datasheet) */
114121f6ecabSRudolf Cornelissen 	/* warning, note:
114221f6ecabSRudolf Cornelissen 	 * datasheet is in error! 60mS needed!! */
114321f6ecabSRudolf Cornelissen 	snooze(60000);
114421f6ecabSRudolf Cornelissen 
114521f6ecabSRudolf Cornelissen 	/* read back updated connection status: */
114621f6ecabSRudolf Cornelissen 	buffer[0] = si->ps.tv_encoder.adress + RD;
114721f6ecabSRudolf Cornelissen 
114821f6ecabSRudolf Cornelissen 	/* transmit 1 byte */
114921f6ecabSRudolf Cornelissen 	i2c_bstart(si->ps.tv_encoder.bus);
115021f6ecabSRudolf Cornelissen 	i2c_writebuffer(si->ps.tv_encoder.bus, buffer, 1);
115121f6ecabSRudolf Cornelissen 
115221f6ecabSRudolf Cornelissen 	/* receive 1 byte */
11539a7218c5SRudolf Cornelissen 	/* ACK level to TX after last byte to RX should be 1 (= NACK) (see I2C spec)
115421f6ecabSRudolf Cornelissen 	 * While the BT's don't care, CX chips will block the SDA line if an ACK gets sent! */
115521f6ecabSRudolf Cornelissen 	buffer[0] = 1;
115621f6ecabSRudolf Cornelissen 	i2c_readbuffer(si->ps.tv_encoder.bus, buffer, 1);
115721f6ecabSRudolf Cornelissen 	i2c_bstop(si->ps.tv_encoder.bus);
115821f6ecabSRudolf Cornelissen 	/* log on errors */
115921f6ecabSRudolf Cornelissen 	stat = i2c_flag_error(0);
116021f6ecabSRudolf Cornelissen 	if (stat)
116121f6ecabSRudolf Cornelissen 	{
116221f6ecabSRudolf Cornelissen 		LOG(4,("Brooktree: I2C errors occurred while reading connection status (3)\n"));
116321f6ecabSRudolf Cornelissen 		return stat;
116421f6ecabSRudolf Cornelissen 	}
116521f6ecabSRudolf Cornelissen 
116621f6ecabSRudolf Cornelissen 	*monstat = ((buffer[0] & 0xe0) >> 5);
116721f6ecabSRudolf Cornelissen 	LOG(4,("Brooktree: TV output monitor status = %d\n", *monstat));
116821f6ecabSRudolf Cornelissen 
116921f6ecabSRudolf Cornelissen 	/* instruct BT to go back to normal operation: */
117021f6ecabSRudolf Cornelissen 	buffer[0] = si->ps.tv_encoder.adress + WR;
117121f6ecabSRudolf Cornelissen 	/* select register with CHECK_STAT CMD */
117221f6ecabSRudolf Cornelissen 	buffer[1] = 0xba;
117321f6ecabSRudolf Cornelissen 	/* issue actual command. */
117421f6ecabSRudolf Cornelissen 	buffer[2] = 0x00;
117521f6ecabSRudolf Cornelissen 
117621f6ecabSRudolf Cornelissen 	i2c_bstart(si->ps.tv_encoder.bus);
117721f6ecabSRudolf Cornelissen 	i2c_writebuffer(si->ps.tv_encoder.bus, buffer, sizeof(buffer));
117821f6ecabSRudolf Cornelissen 	i2c_bstop(si->ps.tv_encoder.bus);
117921f6ecabSRudolf Cornelissen 	/* log on errors */
118021f6ecabSRudolf Cornelissen 	stat = i2c_flag_error(0);
118121f6ecabSRudolf Cornelissen 	if (stat)
118221f6ecabSRudolf Cornelissen 	{
118321f6ecabSRudolf Cornelissen 		LOG(4,("Brooktree: I2C errors occurred while reading connection status (4)\n"));
118421f6ecabSRudolf Cornelissen 		return stat;
118521f6ecabSRudolf Cornelissen 	}
118621f6ecabSRudolf Cornelissen 
118721f6ecabSRudolf Cornelissen 	return stat;
118821f6ecabSRudolf Cornelissen }//end BT_read_monstat.
118921f6ecabSRudolf Cornelissen 
119021f6ecabSRudolf Cornelissen static uint8 BT_killclk_blackout(void)
119121f6ecabSRudolf Cornelissen {
119221f6ecabSRudolf Cornelissen 	uint8 stat;
119321f6ecabSRudolf Cornelissen 
119421f6ecabSRudolf Cornelissen 	uint8 buffer[4];
119521f6ecabSRudolf Cornelissen 
11969a7218c5SRudolf Cornelissen 	LOG(4,("Brooktree: Killing clock and/or blacking out (blocking output signals)\n"));
11979a7218c5SRudolf Cornelissen 
119821f6ecabSRudolf Cornelissen 	/* reset status */
119921f6ecabSRudolf Cornelissen 	i2c_flag_error (-1);
120021f6ecabSRudolf Cornelissen 
120121f6ecabSRudolf Cornelissen 	if (si->ps.tv_encoder.type <= BT869) //BT...
120221f6ecabSRudolf Cornelissen 	{
120321f6ecabSRudolf Cornelissen 		/* Only disable external pixelclock input on BT's.
120421f6ecabSRudolf Cornelissen 		 * CX chips will lock the bus if you do this.
120521f6ecabSRudolf Cornelissen 		 * (It looks like the external pixelclock is always OK as long as a valid
120621f6ecabSRudolf Cornelissen 		 * mode is programmed for the TVout chip. This means that disabling the use
120721f6ecabSRudolf Cornelissen 		 * of this clock is not needed anyway.
120821f6ecabSRudolf Cornelissen 		 * If you do disable this input, this pixelclock will rise to about 60Mhz BTW..) */
120921f6ecabSRudolf Cornelissen 
121021f6ecabSRudolf Cornelissen 		/* disable use of external pixelclock source... */
121121f6ecabSRudolf Cornelissen 		/* (should prevent BT for being 'overclocked' by RIVA in VGA-only mode...) */
121221f6ecabSRudolf Cornelissen 		buffer[0] = si->ps.tv_encoder.adress + WR;
121321f6ecabSRudolf Cornelissen 		/* select BT register for setting EN_XCLK */
121421f6ecabSRudolf Cornelissen 		buffer[1] = 0xa0;
121521f6ecabSRudolf Cornelissen 		/* clear it */
121621f6ecabSRudolf Cornelissen 		buffer[2] = 0x00;
121721f6ecabSRudolf Cornelissen 
121821f6ecabSRudolf Cornelissen 		i2c_bstart(si->ps.tv_encoder.bus);
121921f6ecabSRudolf Cornelissen 		i2c_writebuffer(si->ps.tv_encoder.bus, buffer, 3);
122021f6ecabSRudolf Cornelissen 		i2c_bstop(si->ps.tv_encoder.bus);
122121f6ecabSRudolf Cornelissen 		/* log on errors */
122221f6ecabSRudolf Cornelissen 		stat = i2c_flag_error(0);
122321f6ecabSRudolf Cornelissen 		if (stat)
122421f6ecabSRudolf Cornelissen 		{
122521f6ecabSRudolf Cornelissen 			LOG(4,("Brooktree: I2C errors occurred while doing killclk_blackout (1-BT)\n"));
122621f6ecabSRudolf Cornelissen 			return stat;
122721f6ecabSRudolf Cornelissen 		}
122821f6ecabSRudolf Cornelissen 	}
122921f6ecabSRudolf Cornelissen 	else //CX...
123021f6ecabSRudolf Cornelissen 	{
123121f6ecabSRudolf Cornelissen 		/* Disable CX video out (or wild output will be seen on TV..) */
123221f6ecabSRudolf Cornelissen 		buffer[0] = si->ps.tv_encoder.adress + WR;
123321f6ecabSRudolf Cornelissen 		/* select register in CX */
123421f6ecabSRudolf Cornelissen 		buffer[1] = 0x6c;
123521f6ecabSRudolf Cornelissen 		/* disable active video out. */
123621f6ecabSRudolf Cornelissen 		buffer[2] = 0x02;
123721f6ecabSRudolf Cornelissen 
123821f6ecabSRudolf Cornelissen 		i2c_bstart(si->ps.tv_encoder.bus);
123921f6ecabSRudolf Cornelissen 		i2c_writebuffer(si->ps.tv_encoder.bus, buffer, 3);
124021f6ecabSRudolf Cornelissen 		i2c_bstop(si->ps.tv_encoder.bus);
124121f6ecabSRudolf Cornelissen 		/* log on errors */
124221f6ecabSRudolf Cornelissen 		stat = i2c_flag_error(0);
124321f6ecabSRudolf Cornelissen 		if (stat)
124421f6ecabSRudolf Cornelissen 		{
124521f6ecabSRudolf Cornelissen 			LOG(4,("Brooktree: I2C errors occurred while doing killclk_blackout (1-CX)\n"));
124621f6ecabSRudolf Cornelissen 			return stat;
124721f6ecabSRudolf Cornelissen 		}
124821f6ecabSRudolf Cornelissen 	}
124921f6ecabSRudolf Cornelissen 
125021f6ecabSRudolf Cornelissen 	/* black-out TVout while outputs are enabled... */
125121f6ecabSRudolf Cornelissen 	buffer[0] = si->ps.tv_encoder.adress + WR;
125221f6ecabSRudolf Cornelissen 	/* select first TV config register to write */
125321f6ecabSRudolf Cornelissen 	buffer[1] = 0xc4;
125421f6ecabSRudolf Cornelissen 	/* disable testimage while outputs remain enabled */
125521f6ecabSRudolf Cornelissen 	buffer[2] = 0x01;
125621f6ecabSRudolf Cornelissen 	/* input is 24bit mpx'd RGB, BLANK = in, sync = act. hi */
125721f6ecabSRudolf Cornelissen 	buffer[3] = 0x18;
125821f6ecabSRudolf Cornelissen 
125921f6ecabSRudolf Cornelissen 	i2c_bstart(si->ps.tv_encoder.bus);
126021f6ecabSRudolf Cornelissen 	i2c_writebuffer(si->ps.tv_encoder.bus, buffer, sizeof(buffer));
126121f6ecabSRudolf Cornelissen 	i2c_bstop(si->ps.tv_encoder.bus);
126221f6ecabSRudolf Cornelissen 	/* log on errors */
126321f6ecabSRudolf Cornelissen 	stat = i2c_flag_error(0);
126421f6ecabSRudolf Cornelissen 	if (stat)
126521f6ecabSRudolf Cornelissen 	{
126621f6ecabSRudolf Cornelissen 		LOG(4,("Brooktree: I2C errors occurred while doing killclk_blackout (2)\n"));
126721f6ecabSRudolf Cornelissen 		return stat;
126821f6ecabSRudolf Cornelissen 	}
126921f6ecabSRudolf Cornelissen 
127021f6ecabSRudolf Cornelissen 	return stat;
127121f6ecabSRudolf Cornelissen }//end BT_killclk_blackout.
127221f6ecabSRudolf Cornelissen 
1273b2459715SRudolf Cornelissen uint8 BT_check_tvmode(display_mode target)
1274fb6cadf1Sshatty {
1275b2459715SRudolf Cornelissen 	uint8 status = NOT_SUPPORTED;
1276b2459715SRudolf Cornelissen 	uint32 mode = ((target.timing.h_display) | ((target.timing.v_display) << 16));
1277b2459715SRudolf Cornelissen 
1278b2459715SRudolf Cornelissen 	switch (mode)
1279b2459715SRudolf Cornelissen 	{
1280b2459715SRudolf Cornelissen 	case (640 | (480 << 16)):
1281*c5fe5f2dSRudolf Cornelissen 		if (((target.flags & TV_BITS) == TV_PAL) && (!(target.flags & TV_VIDEO)))
1282*c5fe5f2dSRudolf Cornelissen 			status = PAL640;
1283b2459715SRudolf Cornelissen 		if ((target.flags & TV_BITS) == TV_NTSC)
1284b2459715SRudolf Cornelissen 		{
1285b2459715SRudolf Cornelissen 			if (!(target.flags & TV_VIDEO)) status = NTSC640;
1286b2459715SRudolf Cornelissen 			else status = NTSC640_OS;
1287b2459715SRudolf Cornelissen 		}
1288b2459715SRudolf Cornelissen 		break;
1289*c5fe5f2dSRudolf Cornelissen 	case (768 | (576 << 16)):
1290*c5fe5f2dSRudolf Cornelissen 		if (((target.flags & TV_BITS) == TV_PAL) && (target.flags & TV_VIDEO))
1291*c5fe5f2dSRudolf Cornelissen 			status = PAL800_OS;
1292*c5fe5f2dSRudolf Cornelissen 		break;
1293b2459715SRudolf Cornelissen 	case (800 | (600 << 16)):
1294*c5fe5f2dSRudolf Cornelissen 		if (((target.flags & TV_BITS) == TV_PAL) && (!(target.flags & TV_VIDEO)))
1295*c5fe5f2dSRudolf Cornelissen 			status = PAL800;
1296*c5fe5f2dSRudolf Cornelissen 		if (((target.flags & TV_BITS) == TV_NTSC) && (!(target.flags & TV_VIDEO)))
1297*c5fe5f2dSRudolf Cornelissen 			status = NTSC800;
1298b2459715SRudolf Cornelissen 		break;
1299b2459715SRudolf Cornelissen 	case (720 | (480 << 16)):
1300b2459715SRudolf Cornelissen 		if (((target.flags & TV_BITS) == TV_NTSC) && (target.flags & TV_VIDEO))
1301b2459715SRudolf Cornelissen 			status = NTSC720;
1302b2459715SRudolf Cornelissen 		break;
1303b2459715SRudolf Cornelissen 	case (720 | (576 << 16)):
1304b2459715SRudolf Cornelissen 		if (((target.flags & TV_BITS) == TV_PAL) && (target.flags & TV_VIDEO))
1305b2459715SRudolf Cornelissen 			status = PAL720;
1306b2459715SRudolf Cornelissen 		break;
1307b2459715SRudolf Cornelissen 	}
1308b2459715SRudolf Cornelissen 
1309b2459715SRudolf Cornelissen 	return status;
1310b2459715SRudolf Cornelissen }//end BT_check_tvmode.
1311b2459715SRudolf Cornelissen 
1312b2459715SRudolf Cornelissen status_t BT_setmode(display_mode target)
1313b2459715SRudolf Cornelissen {
1314efe9c7a6SRudolf Cornelissen 	uint8 tvmode, monstat;
1315efe9c7a6SRudolf Cornelissen 	/* enable flickerfilter in desktop modes, disable it in video modes. */
1316efe9c7a6SRudolf Cornelissen 	uint8 ffilter = 0;
1317b2459715SRudolf Cornelissen 
1318fb6cadf1Sshatty 	/* use a display_mode copy because we might tune it for TVout compatibility */
1319fb6cadf1Sshatty 	display_mode tv_target = target;
1320fb6cadf1Sshatty 
1321fb6cadf1Sshatty 	/* preset new TVout mode */
1322b2459715SRudolf Cornelissen 	tvmode = BT_check_tvmode(tv_target);
1323b2459715SRudolf Cornelissen 	if (!tvmode) return B_ERROR;
1324b2459715SRudolf Cornelissen 
1325b2459715SRudolf Cornelissen //fixme: only testing singlehead cards for now...
1326b2459715SRudolf Cornelissen if (si->ps.secondary_head)
1327fb6cadf1Sshatty {
1328b2459715SRudolf Cornelissen 	head1_set_timing(tv_target);
1329b2459715SRudolf Cornelissen 	return B_ERROR;
1330fb6cadf1Sshatty }
1331b2459715SRudolf Cornelissen 
1332efe9c7a6SRudolf Cornelissen 	/* fixme: reset BT should be here...
1333efe9c7a6SRudolf Cornelissen 	 * (but: beware of the 'locked SDA' syndrome then!) */
1334efe9c7a6SRudolf Cornelissen 	BT_killclk_blackout();
1335efe9c7a6SRudolf Cornelissen 
1336efe9c7a6SRudolf Cornelissen 	/* read current output devices connection status */
1337efe9c7a6SRudolf Cornelissen 	BT_read_monstat(&monstat);
1338efe9c7a6SRudolf Cornelissen 
1339efe9c7a6SRudolf Cornelissen 	//fixme?
1340efe9c7a6SRudolf Cornelissen 	//'slowdown RIVA clock' if TVout is requested:
1341efe9c7a6SRudolf Cornelissen 	//makes sure TVout chip is not being overclocked! (VESA640/800 is safe..)
1342efe9c7a6SRudolf Cornelissen 	//NOTE: it looks like this is unnesessary after all..
1343efe9c7a6SRudolf Cornelissen 
1344efe9c7a6SRudolf Cornelissen 	/* (pre)set TV mode */
1345efe9c7a6SRudolf Cornelissen 	/* note:
1346efe9c7a6SRudolf Cornelissen 	 * Manual config is non-dependent of the state of the PAL hardware input pin;
1347efe9c7a6SRudolf Cornelissen 	 * Also SDA lockups occur when setting EN_XCLK after autoconfig!
1348efe9c7a6SRudolf Cornelissen 	 * Make sure PAL_MD=0 for NTSC and PAL_MD = 1 for PAL... */
1349b2459715SRudolf Cornelissen 	switch (tvmode)
1350fb6cadf1Sshatty 	{
1351b2459715SRudolf Cornelissen 	case NTSC640:
1352b2459715SRudolf Cornelissen 	case NTSC640_TST:
1353efe9c7a6SRudolf Cornelissen 		ffilter = 1;
1354efe9c7a6SRudolf Cornelissen 		BT_init_NTSC640();
1355b2459715SRudolf Cornelissen 		break;
1356b2459715SRudolf Cornelissen 	case NTSC800:
1357efe9c7a6SRudolf Cornelissen 		ffilter = 1;
1358efe9c7a6SRudolf Cornelissen 		BT_init_NTSC800();
1359b2459715SRudolf Cornelissen 		break;
1360b2459715SRudolf Cornelissen 	case PAL640:
1361efe9c7a6SRudolf Cornelissen 		ffilter = 1;
1362efe9c7a6SRudolf Cornelissen 		BT_init_PAL640();
1363b2459715SRudolf Cornelissen 		break;
1364b2459715SRudolf Cornelissen 	case PAL800:
1365b2459715SRudolf Cornelissen 	case PAL800_TST:
1366efe9c7a6SRudolf Cornelissen 		ffilter = 1;
1367efe9c7a6SRudolf Cornelissen 		BT_init_PAL800();
1368b2459715SRudolf Cornelissen 		break;
1369b2459715SRudolf Cornelissen 	case NTSC640_OS:
1370efe9c7a6SRudolf Cornelissen 		BT_init_NTSC640_OS();
1371b2459715SRudolf Cornelissen 		break;
1372b2459715SRudolf Cornelissen 	case PAL800_OS:
1373efe9c7a6SRudolf Cornelissen 		BT_init_PAL800_OS();
1374b2459715SRudolf Cornelissen 		break;
1375b2459715SRudolf Cornelissen 	case NTSC720:
1376efe9c7a6SRudolf Cornelissen 		BT_init_NTSC720();
1377b2459715SRudolf Cornelissen 		break;
1378b2459715SRudolf Cornelissen 	case PAL720:
1379efe9c7a6SRudolf Cornelissen 		BT_init_PAL720();
1380b2459715SRudolf Cornelissen 		break;
1381fb6cadf1Sshatty 	}
1382fb6cadf1Sshatty 
1383efe9c7a6SRudolf Cornelissen 	/* modify BT Hphase signal to center TV image... */
1384efe9c7a6SRudolf Cornelissen 	BT_setup_hphase(tvmode);
1385fb6cadf1Sshatty 
1386efe9c7a6SRudolf Cornelissen 	/* disable Macro mode */
1387efe9c7a6SRudolf Cornelissen 	switch (tvmode)
1388efe9c7a6SRudolf Cornelissen 	{
1389efe9c7a6SRudolf Cornelissen 	case NTSC640:
1390efe9c7a6SRudolf Cornelissen 	case NTSC640_TST:
1391efe9c7a6SRudolf Cornelissen 	case NTSC800:
1392efe9c7a6SRudolf Cornelissen 	case NTSC640_OS:
1393efe9c7a6SRudolf Cornelissen 	case NTSC720:
1394efe9c7a6SRudolf Cornelissen 		/* NTSC */
1395efe9c7a6SRudolf Cornelissen 		BT_set_macro (0, 0);
1396efe9c7a6SRudolf Cornelissen 		break;
1397efe9c7a6SRudolf Cornelissen 	default:
1398efe9c7a6SRudolf Cornelissen 		/* PAL */
1399efe9c7a6SRudolf Cornelissen 		BT_set_macro (1, 0);
1400efe9c7a6SRudolf Cornelissen 		break;
1401efe9c7a6SRudolf Cornelissen 	}
1402fb6cadf1Sshatty 
1403efe9c7a6SRudolf Cornelissen 	/* setup output signal routing and flickerfilter */
1404efe9c7a6SRudolf Cornelissen 	//fixme: add output force settings in nv.settings, defaulting to autodetect.
1405efe9c7a6SRudolf Cornelissen 	BT_setup_output(monstat, 0, ffilter);
1406fb6cadf1Sshatty 
1407efe9c7a6SRudolf Cornelissen //tmp: enabling testimage...
1408efe9c7a6SRudolf Cornelissen BT_testsignal();
1409fb6cadf1Sshatty 
1410efe9c7a6SRudolf Cornelissen 	//fixme: add custom fixed modelines here that will be pgm'd into the CRTC...
1411fb6cadf1Sshatty 
1412efe9c7a6SRudolf Cornelissen 	/* setup GPU CRTC timing */
1413b2459715SRudolf Cornelissen 	head1_set_timing(tv_target);
1414fb6cadf1Sshatty 
1415efe9c7a6SRudolf Cornelissen 	//fixme: set GPU CRTC to slave mode...
1416efe9c7a6SRudolf Cornelissen 
1417efe9c7a6SRudolf Cornelissen 	//fixme: add code to disable VGA screen when TVout enabled
1418efe9c7a6SRudolf Cornelissen 	//(use via nv.setting preset)
1419fb6cadf1Sshatty 
1420b2459715SRudolf Cornelissen 	return B_OK;
1421fb6cadf1Sshatty }
1422