xref: /plan9/sys/src/cmd/aux/vga/s3generic.c (revision fb7f0c934c48abaed6040d054ef636408c3c522d)
1219b2ee8SDavid du Colombier #include <u.h>
2219b2ee8SDavid du Colombier #include <libc.h>
37dd7cddfSDavid du Colombier #include <bio.h>
4219b2ee8SDavid du Colombier 
59a747e4fSDavid du Colombier #include "pci.h"
6219b2ee8SDavid du Colombier #include "vga.h"
7219b2ee8SDavid du Colombier 
8219b2ee8SDavid du Colombier /*
9219b2ee8SDavid du Colombier  * Generic S3 GUI Accelerator.
10219b2ee8SDavid du Colombier  */
11219b2ee8SDavid du Colombier static void
snarf(Vga * vga,Ctlr * ctlr)127dd7cddfSDavid du Colombier snarf(Vga* vga, Ctlr* ctlr)
13219b2ee8SDavid du Colombier {
147dd7cddfSDavid du Colombier 	int i;
157dd7cddfSDavid du Colombier 
167dd7cddfSDavid du Colombier 	trace("%s->snarf->s3generic\n", ctlr->name);
177dd7cddfSDavid du Colombier 
18219b2ee8SDavid du Colombier 	/*
197dd7cddfSDavid du Colombier 	 * Unlock extended registers.
20219b2ee8SDavid du Colombier 	 * 0xA5 ensures Crt36 and Crt37 are also unlocked
21219b2ee8SDavid du Colombier 	 * (0xA0 unlocks everything else).
22219b2ee8SDavid du Colombier 	 */
23219b2ee8SDavid du Colombier 	vgaxo(Crtx, 0x38, 0x48);
24219b2ee8SDavid du Colombier 	vgaxo(Crtx, 0x39, 0xA5);
25219b2ee8SDavid du Colombier 
26219b2ee8SDavid du Colombier 	/*
27219b2ee8SDavid du Colombier 	 * Not all registers exist on all chips.
28219b2ee8SDavid du Colombier 	 * Crt3[EF] don't exist on any.
29219b2ee8SDavid du Colombier 	 */
30219b2ee8SDavid du Colombier 	for(i = 0x30; i < 0x70; i++)
31219b2ee8SDavid du Colombier 		vga->crt[i] = vgaxi(Crtx, i);
32219b2ee8SDavid du Colombier 
33219b2ee8SDavid du Colombier 	/*
34219b2ee8SDavid du Colombier 	 * Memory size.
35219b2ee8SDavid du Colombier 	 */
36219b2ee8SDavid du Colombier 	switch((vga->crt[0x36]>>5) & 0x07){
37219b2ee8SDavid du Colombier 
38219b2ee8SDavid du Colombier 	case 0x00:
397dd7cddfSDavid du Colombier 		vga->vmz = 4*1024*1024;
40219b2ee8SDavid du Colombier 		break;
41219b2ee8SDavid du Colombier 
42219b2ee8SDavid du Colombier 	case 0x02:
437dd7cddfSDavid du Colombier 		vga->vmz = 3*1024*1024;
44219b2ee8SDavid du Colombier 		break;
45219b2ee8SDavid du Colombier 
46219b2ee8SDavid du Colombier 	case 0x04:
477dd7cddfSDavid du Colombier 		vga->vmz = 2*1024*1024;
48219b2ee8SDavid du Colombier 		break;
49219b2ee8SDavid du Colombier 
50219b2ee8SDavid du Colombier 	case 0x06:
517dd7cddfSDavid du Colombier 		vga->vmz = 1*1024*1024;
52219b2ee8SDavid du Colombier 		break;
53219b2ee8SDavid du Colombier 
54219b2ee8SDavid du Colombier 	case 0x07:
557dd7cddfSDavid du Colombier 		vga->vmz = 512*1024;
56219b2ee8SDavid du Colombier 		break;
57219b2ee8SDavid du Colombier 	}
58219b2ee8SDavid du Colombier 
59219b2ee8SDavid du Colombier 	ctlr->flag |= Fsnarf;
60219b2ee8SDavid du Colombier }
61219b2ee8SDavid du Colombier 
62219b2ee8SDavid du Colombier static void
init(Vga * vga,Ctlr * ctlr)63219b2ee8SDavid du Colombier init(Vga* vga, Ctlr* ctlr)
64219b2ee8SDavid du Colombier {
65219b2ee8SDavid du Colombier 	Mode *mode;
66219b2ee8SDavid du Colombier 	ulong x;
67219b2ee8SDavid du Colombier 
687dd7cddfSDavid du Colombier 	trace("%s->init->s3generic\n", ctlr->name);
69219b2ee8SDavid du Colombier 	mode = vga->mode;
70219b2ee8SDavid du Colombier 
71219b2ee8SDavid du Colombier 	/*
727dd7cddfSDavid du Colombier 	 * Is enhanced mode is necessary?
73219b2ee8SDavid du Colombier 	 */
74219b2ee8SDavid du Colombier 	if((ctlr->flag & (Uenhanced|Henhanced)) == Henhanced){
757dd7cddfSDavid du Colombier 		if(mode->z >= 8)
76219b2ee8SDavid du Colombier 			resyncinit(vga, ctlr, Uenhanced, 0);
77219b2ee8SDavid du Colombier 		else
78219b2ee8SDavid du Colombier 			resyncinit(vga, ctlr, 0, Uenhanced|Henhanced);
79219b2ee8SDavid du Colombier 	}
807dd7cddfSDavid du Colombier 	if((ctlr->flag & Uenhanced) == 0 && mode->x > 1024)
817dd7cddfSDavid du Colombier 		error("%s: no support for 1-bit mode > 1024x768x1\n", ctlr->name);
82219b2ee8SDavid du Colombier 
83219b2ee8SDavid du Colombier 	vga->crt[0x31] = 0x85;
847dd7cddfSDavid du Colombier 	vga->crt[0x6A] &= 0xC0;
85219b2ee8SDavid du Colombier 	vga->crt[0x32] &= ~0x40;
86219b2ee8SDavid du Colombier 
87219b2ee8SDavid du Colombier 	vga->crt[0x31] |= 0x08;
88219b2ee8SDavid du Colombier 	vga->crt[0x32] |= 0x40;
89219b2ee8SDavid du Colombier 
90219b2ee8SDavid du Colombier 	vga->crt[0x33] |= 0x20;
917dd7cddfSDavid du Colombier 	if(mode->z >= 8)
92219b2ee8SDavid du Colombier 		vga->crt[0x3A] |= 0x10;
93219b2ee8SDavid du Colombier 	else
94219b2ee8SDavid du Colombier 		vga->crt[0x3A] &= ~0x10;
95219b2ee8SDavid du Colombier 
96219b2ee8SDavid du Colombier 	vga->crt[0x34] = 0x10;
97219b2ee8SDavid du Colombier 	vga->crt[0x35] = 0x00;
98219b2ee8SDavid du Colombier 	if(mode->interlace){
99219b2ee8SDavid du Colombier 		vga->crt[0x3C] = vga->crt[0]/2;
100219b2ee8SDavid du Colombier 		vga->crt[0x42] |= 0x20;
101219b2ee8SDavid du Colombier 	}
102219b2ee8SDavid du Colombier 	else{
103219b2ee8SDavid du Colombier 		vga->crt[0x3C] = 0x00;
104219b2ee8SDavid du Colombier 		vga->crt[0x42] &= ~0x20;
105219b2ee8SDavid du Colombier 	}
106219b2ee8SDavid du Colombier 
107219b2ee8SDavid du Colombier 	vga->crt[0x40] = (vga->crt[0x40] & 0xF2);
108219b2ee8SDavid du Colombier 	vga->crt[0x43] = 0x00;
1097dd7cddfSDavid du Colombier 	vga->crt[0x45] = 0x00;
110219b2ee8SDavid du Colombier 
111219b2ee8SDavid du Colombier 	vga->crt[0x50] &= 0x3E;
112219b2ee8SDavid du Colombier 	if(mode->x <= 640)
113219b2ee8SDavid du Colombier 		x = 0x40;
114219b2ee8SDavid du Colombier 	else if(mode->x <= 800)
115219b2ee8SDavid du Colombier 		x = 0x80;
116219b2ee8SDavid du Colombier 	else if(mode->x <= 1024)
117219b2ee8SDavid du Colombier 		x = 0x00;
118219b2ee8SDavid du Colombier 	else if(mode->x <= 1152)
119219b2ee8SDavid du Colombier 		x = 0x01;
120219b2ee8SDavid du Colombier 	else if(mode->x <= 1280)
121219b2ee8SDavid du Colombier 		x = 0xC0;
1227dd7cddfSDavid du Colombier 	else
123219b2ee8SDavid du Colombier 		x = 0x81;
124219b2ee8SDavid du Colombier 	vga->crt[0x50] |= x;
125219b2ee8SDavid du Colombier 
126219b2ee8SDavid du Colombier 	vga->crt[0x51] = (vga->crt[0x51] & 0xC3)|((vga->crt[0x13]>>4) & 0x30);
127219b2ee8SDavid du Colombier 	vga->crt[0x53] &= ~0x10;
128219b2ee8SDavid du Colombier 
129219b2ee8SDavid du Colombier 	/*
1307dd7cddfSDavid du Colombier 	 * Set up linear aperture. For the moment it's 64K at 0xA0000.
1317dd7cddfSDavid du Colombier 	 * The real base address will be assigned before load is called.
132219b2ee8SDavid du Colombier 	 */
133219b2ee8SDavid du Colombier 	vga->crt[0x58] = 0x88;
1347dd7cddfSDavid du Colombier 	if(ctlr->flag & Uenhanced){
135219b2ee8SDavid du Colombier 		vga->crt[0x58] |= 0x10;
1367dd7cddfSDavid du Colombier 		if(vga->linear && (ctlr->flag & Hlinear))
1377dd7cddfSDavid du Colombier 			ctlr->flag |= Ulinear;
1387dd7cddfSDavid du Colombier 		if(vga->vmz <= 1024*1024)
1397dd7cddfSDavid du Colombier 			vga->vma = 1024*1024;
1407dd7cddfSDavid du Colombier 		else if(vga->vmz <= 2*1024*1024)
1417dd7cddfSDavid du Colombier 			vga->vma = 2*1024*1024;
1427dd7cddfSDavid du Colombier 		else
1437dd7cddfSDavid du Colombier 			vga->vma = 8*1024*1024;
1447dd7cddfSDavid du Colombier 	}
145219b2ee8SDavid du Colombier 	vga->crt[0x59] = 0x00;
146219b2ee8SDavid du Colombier 	vga->crt[0x5A] = 0x0A;
147219b2ee8SDavid du Colombier 
148219b2ee8SDavid du Colombier 	vga->crt[0x5D] &= 0x80;
149219b2ee8SDavid du Colombier 	if(vga->crt[0x00] & 0x100)
150219b2ee8SDavid du Colombier 		vga->crt[0x5D] |= 0x01;
151219b2ee8SDavid du Colombier 	if(vga->crt[0x01] & 0x100)
152219b2ee8SDavid du Colombier 		vga->crt[0x5D] |= 0x02;
153219b2ee8SDavid du Colombier 	if(vga->crt[0x02] & 0x100)
154219b2ee8SDavid du Colombier 		vga->crt[0x5D] |= 0x04;
155219b2ee8SDavid du Colombier 	if(vga->crt[0x04] & 0x100)
156219b2ee8SDavid du Colombier 		vga->crt[0x5D] |= 0x10;
157219b2ee8SDavid du Colombier 	if(vga->crt[0x3B] & 0x100)
158219b2ee8SDavid du Colombier 		vga->crt[0x5D] |= 0x40;
159219b2ee8SDavid du Colombier 
160219b2ee8SDavid du Colombier 	vga->crt[0x5E] = 0x40;
161219b2ee8SDavid du Colombier 	if(vga->crt[0x06] & 0x400)
162219b2ee8SDavid du Colombier 		vga->crt[0x5E] |= 0x01;
163219b2ee8SDavid du Colombier 	if(vga->crt[0x12] & 0x400)
164219b2ee8SDavid du Colombier 		vga->crt[0x5E] |= 0x02;
165219b2ee8SDavid du Colombier 	if(vga->crt[0x15] & 0x400)
166219b2ee8SDavid du Colombier 		vga->crt[0x5E] |= 0x04;
167219b2ee8SDavid du Colombier 	if(vga->crt[0x10] & 0x400)
168219b2ee8SDavid du Colombier 		vga->crt[0x5E] |= 0x10;
169219b2ee8SDavid du Colombier 
170219b2ee8SDavid du Colombier 	ctlr->type = s3generic.name;
171219b2ee8SDavid du Colombier 
172219b2ee8SDavid du Colombier 	ctlr->flag |= Finit;
173219b2ee8SDavid du Colombier }
174219b2ee8SDavid du Colombier 
175219b2ee8SDavid du Colombier static void
load(Vga * vga,Ctlr * ctlr)176219b2ee8SDavid du Colombier load(Vga* vga, Ctlr* ctlr)
177219b2ee8SDavid du Colombier {
1787dd7cddfSDavid du Colombier 	ulong l;
1797dd7cddfSDavid du Colombier 
1807dd7cddfSDavid du Colombier 	trace("%s->load->s3generic\n", ctlr->name);
181219b2ee8SDavid du Colombier 
182219b2ee8SDavid du Colombier 	vgaxo(Crtx, 0x31, vga->crt[0x31]);
183219b2ee8SDavid du Colombier 	vgaxo(Crtx, 0x32, vga->crt[0x32]);
184219b2ee8SDavid du Colombier 	vgaxo(Crtx, 0x33, vga->crt[0x33]);
185219b2ee8SDavid du Colombier 	vgaxo(Crtx, 0x34, vga->crt[0x34]);
186219b2ee8SDavid du Colombier 	vgaxo(Crtx, 0x35, vga->crt[0x35]);
187219b2ee8SDavid du Colombier 	vgaxo(Crtx, 0x3A, vga->crt[0x3A]);
188219b2ee8SDavid du Colombier 	vgaxo(Crtx, 0x3B, vga->crt[0x3B]);
189219b2ee8SDavid du Colombier 	vgaxo(Crtx, 0x3C, vga->crt[0x3C]);
190219b2ee8SDavid du Colombier 
191219b2ee8SDavid du Colombier 	vgaxo(Crtx, 0x40, vga->crt[0x40]|0x01);
192219b2ee8SDavid du Colombier 	vgaxo(Crtx, 0x42, vga->crt[0x42]);
193219b2ee8SDavid du Colombier 	vgaxo(Crtx, 0x43, vga->crt[0x43]);
194219b2ee8SDavid du Colombier 	vgaxo(Crtx, 0x45, vga->crt[0x45]);
195219b2ee8SDavid du Colombier 
196219b2ee8SDavid du Colombier 	vgaxo(Crtx, 0x50, vga->crt[0x50]);
197219b2ee8SDavid du Colombier 	vgaxo(Crtx, 0x51, vga->crt[0x51]);
198219b2ee8SDavid du Colombier 	vgaxo(Crtx, 0x53, vga->crt[0x53]);
199219b2ee8SDavid du Colombier 	vgaxo(Crtx, 0x54, vga->crt[0x54]);
200219b2ee8SDavid du Colombier 	vgaxo(Crtx, 0x55, vga->crt[0x55]);
2017dd7cddfSDavid du Colombier 
2027dd7cddfSDavid du Colombier 	if(ctlr->flag & Ulinear){
2037dd7cddfSDavid du Colombier 		l = vga->vmb>>16;
2047dd7cddfSDavid du Colombier 		vga->crt[0x59] = (l>>8) & 0xFF;
2057dd7cddfSDavid du Colombier 		vga->crt[0x5A] = l & 0xFF;
2067dd7cddfSDavid du Colombier 		if(vga->vmz <= 1024*1024)
2077dd7cddfSDavid du Colombier 			vga->crt[0x58] |= 0x01;
2087dd7cddfSDavid du Colombier 		else if(vga->vmz <= 2*1024*1024)
2097dd7cddfSDavid du Colombier 			vga->crt[0x58] |= 0x02;
2107dd7cddfSDavid du Colombier 		else
2117dd7cddfSDavid du Colombier 			vga->crt[0x58] |= 0x03;
2127dd7cddfSDavid du Colombier 	}
213219b2ee8SDavid du Colombier 	vgaxo(Crtx, 0x59, vga->crt[0x59]);
214219b2ee8SDavid du Colombier 	vgaxo(Crtx, 0x5A, vga->crt[0x5A]);
215219b2ee8SDavid du Colombier 	vgaxo(Crtx, 0x58, vga->crt[0x58]);
2167dd7cddfSDavid du Colombier 
217219b2ee8SDavid du Colombier 	vgaxo(Crtx, 0x5D, vga->crt[0x5D]);
218219b2ee8SDavid du Colombier 	vgaxo(Crtx, 0x5E, vga->crt[0x5E]);
219219b2ee8SDavid du Colombier 
2207dd7cddfSDavid du Colombier 	vgaxo(Crtx, 0x6A, vga->crt[0x6A]);
2217dd7cddfSDavid du Colombier 
222219b2ee8SDavid du Colombier 	ctlr->flag |= Fload;
223219b2ee8SDavid du Colombier }
224219b2ee8SDavid du Colombier 
225219b2ee8SDavid du Colombier static void
dump(Vga * vga,Ctlr * ctlr)226219b2ee8SDavid du Colombier dump(Vga* vga, Ctlr* ctlr)
227219b2ee8SDavid du Colombier {
228*fb7f0c93SDavid du Colombier 	int i, id, interlace, mul, div;
229219b2ee8SDavid du Colombier 	char *name;
230219b2ee8SDavid du Colombier 	ushort shb, vrs, x;
231219b2ee8SDavid du Colombier 
232219b2ee8SDavid du Colombier 	name = ctlr->name;
233219b2ee8SDavid du Colombier 
234219b2ee8SDavid du Colombier 	printitem(name, "Crt30");
235219b2ee8SDavid du Colombier 	for(i = 0x30; i < 0x3E; i++)
236219b2ee8SDavid du Colombier 		printreg(vga->crt[i]);
237219b2ee8SDavid du Colombier 
238219b2ee8SDavid du Colombier 	printitem(name, "Crt40");
239219b2ee8SDavid du Colombier 	for(i = 0x40; i < 0x50; i++)
240219b2ee8SDavid du Colombier 		printreg(vga->crt[i]);
241219b2ee8SDavid du Colombier 
242219b2ee8SDavid du Colombier 	printitem(name, "Crt50");
243219b2ee8SDavid du Colombier 	for(i = 0x50; i < 0x60; i++)
244219b2ee8SDavid du Colombier 		printreg(vga->crt[i]);
245219b2ee8SDavid du Colombier 
246219b2ee8SDavid du Colombier 	printitem(name, "Crt60");
247219b2ee8SDavid du Colombier 	for(i = 0x60; i < 0x70; i++)
248219b2ee8SDavid du Colombier 		printreg(vga->crt[i]);
249219b2ee8SDavid du Colombier 
250219b2ee8SDavid du Colombier 	/*
251219b2ee8SDavid du Colombier 	 * Try to disassemble the snarfed values into
252219b2ee8SDavid du Colombier 	 * understandable numbers.
253219b2ee8SDavid du Colombier 	 * Only do this if we weren't called after Finit.
254219b2ee8SDavid du Colombier 	 */
255219b2ee8SDavid du Colombier 	if(ctlr->flag & Finit)
256219b2ee8SDavid du Colombier 		return;
257219b2ee8SDavid du Colombier 
258*fb7f0c93SDavid du Colombier 
259219b2ee8SDavid du Colombier 	/*
260219b2ee8SDavid du Colombier 	 * If hde <= 400, assume this is a 928 or Vision964
2617dd7cddfSDavid du Colombier 	 * and the horizontal values have been divided by 4.
262*fb7f0c93SDavid du Colombier 	 *
263*fb7f0c93SDavid du Colombier 	 * if ViRGE/[DG]X and 15 or 16bpp, horizontal values have
264*fb7f0c93SDavid du Colombier 	 * been multiplied by 2.
265219b2ee8SDavid du Colombier 	 */
266219b2ee8SDavid du Colombier 	mul = 1;
267*fb7f0c93SDavid du Colombier 	div = 1;
268*fb7f0c93SDavid du Colombier 
269*fb7f0c93SDavid du Colombier 	if(strcmp(name, "virge") == 0){
270*fb7f0c93SDavid du Colombier 		id = (vga->crt[0x2D]<<8)|vga->crt[0x2E];
271*fb7f0c93SDavid du Colombier 		/* S3 ViRGE/[DG]X */
272*fb7f0c93SDavid du Colombier 		if(id==0x8A01 && ((vga->crt[0x67] & 0x30) || (vga->crt[0x67] & 0x50))){
273*fb7f0c93SDavid du Colombier 			mul = 1;
274*fb7f0c93SDavid du Colombier 			div = 2;
275*fb7f0c93SDavid du Colombier 		}
276*fb7f0c93SDavid du Colombier 	}
277*fb7f0c93SDavid du Colombier 
278219b2ee8SDavid du Colombier 	x = vga->crt[0x01];
279219b2ee8SDavid du Colombier 	if(vga->crt[0x5D] & 0x02)
280219b2ee8SDavid du Colombier 		x |= 0x100;
281219b2ee8SDavid du Colombier 	x = (x+1)<<3;
282*fb7f0c93SDavid du Colombier 
283*fb7f0c93SDavid du Colombier 	if(x <= 400){
284219b2ee8SDavid du Colombier 		mul = 4;
285*fb7f0c93SDavid du Colombier 		div = 1;
286*fb7f0c93SDavid du Colombier 	}
287*fb7f0c93SDavid du Colombier 
288*fb7f0c93SDavid du Colombier 	x = (x * mul) / div;
289219b2ee8SDavid du Colombier 	printitem(name, "hde");
290219b2ee8SDavid du Colombier 	printreg(x);
2917dd7cddfSDavid du Colombier 	Bprint(&stdout, "%6ud", x);
292219b2ee8SDavid du Colombier 
293219b2ee8SDavid du Colombier 	shb = vga->crt[0x02];
294219b2ee8SDavid du Colombier 	if(vga->crt[0x5D] & 0x04)
295219b2ee8SDavid du Colombier 		shb |= 0x100;
296219b2ee8SDavid du Colombier 	shb = (shb+1)<<3;
297*fb7f0c93SDavid du Colombier 	shb = (shb * mul) / div;
298219b2ee8SDavid du Colombier 	printitem(name, "shb");
299219b2ee8SDavid du Colombier 	printreg(shb);
3007dd7cddfSDavid du Colombier 	Bprint(&stdout, "%6ud", shb);
301219b2ee8SDavid du Colombier 
302219b2ee8SDavid du Colombier 	x = vga->crt[0x03] & 0x1F;
303219b2ee8SDavid du Colombier 	if(vga->crt[0x05] & 0x80)
304219b2ee8SDavid du Colombier 		x |= 0x20;
305*fb7f0c93SDavid du Colombier 	x = (x * mul) / div;
306219b2ee8SDavid du Colombier 	x = shb|x;					/* ???? */
307219b2ee8SDavid du Colombier 	if(vga->crt[0x5D] & 0x08)
308219b2ee8SDavid du Colombier 		x += 64;
309219b2ee8SDavid du Colombier 	printitem(name, "ehb");
310219b2ee8SDavid du Colombier 	printreg(x);
3117dd7cddfSDavid du Colombier 	Bprint(&stdout, "%6ud", x);
312219b2ee8SDavid du Colombier 
313219b2ee8SDavid du Colombier 	x = vga->crt[0x00];
314219b2ee8SDavid du Colombier 	if(vga->crt[0x5D] & 0x01)
315219b2ee8SDavid du Colombier 		x |= 0x100;
316219b2ee8SDavid du Colombier 	x = (x+5)<<3;
317*fb7f0c93SDavid du Colombier 	x = (x * mul) / div;
318219b2ee8SDavid du Colombier 	printitem(name, "ht");
319219b2ee8SDavid du Colombier 	printreg(x);
3207dd7cddfSDavid du Colombier 	Bprint(&stdout, "%6ud", x);
321219b2ee8SDavid du Colombier 
322219b2ee8SDavid du Colombier 	interlace = vga->crt[0x42] & 0x20;
323219b2ee8SDavid du Colombier 	x = vga->crt[0x12];
324219b2ee8SDavid du Colombier 	if(vga->crt[0x07] & 0x02)
325219b2ee8SDavid du Colombier 		x |= 0x100;
326219b2ee8SDavid du Colombier 	if(vga->crt[0x07] & 0x40)
327219b2ee8SDavid du Colombier 		x |= 0x200;
328219b2ee8SDavid du Colombier 	if(vga->crt[0x5E] & 0x02)
329219b2ee8SDavid du Colombier 		x |= 0x400;
330219b2ee8SDavid du Colombier 	x += 1;
331219b2ee8SDavid du Colombier 	if(interlace)
332219b2ee8SDavid du Colombier 		x *= 2;
333219b2ee8SDavid du Colombier 	printitem(name, "vde");
334219b2ee8SDavid du Colombier 	printreg(x);
3357dd7cddfSDavid du Colombier 	Bprint(&stdout, "%6ud", x);
336219b2ee8SDavid du Colombier 
337219b2ee8SDavid du Colombier 	vrs = vga->crt[0x10];
338219b2ee8SDavid du Colombier 	if(vga->crt[0x07] & 0x04)
339219b2ee8SDavid du Colombier 		vrs |= 0x100;
340219b2ee8SDavid du Colombier 	if(vga->crt[0x07] & 0x80)
341219b2ee8SDavid du Colombier 		vrs |= 0x200;
342219b2ee8SDavid du Colombier 	if(vga->crt[0x5E] & 0x10)
343219b2ee8SDavid du Colombier 		vrs |= 0x400;
344219b2ee8SDavid du Colombier 	if(interlace)
345219b2ee8SDavid du Colombier 		vrs *= 2;
346219b2ee8SDavid du Colombier 	printitem(name, "vrs");
347219b2ee8SDavid du Colombier 	printreg(vrs);
3487dd7cddfSDavid du Colombier 	Bprint(&stdout, "%6ud", vrs);
349219b2ee8SDavid du Colombier 
350219b2ee8SDavid du Colombier 	if(interlace)
351219b2ee8SDavid du Colombier 		vrs /= 2;
352219b2ee8SDavid du Colombier 	x = (vrs & ~0x0F)|(vga->crt[0x11] & 0x0F);
353219b2ee8SDavid du Colombier 	if(interlace)
354219b2ee8SDavid du Colombier 		x *= 2;
355219b2ee8SDavid du Colombier 	printitem(name, "vre");
356219b2ee8SDavid du Colombier 	printreg(x);
3577dd7cddfSDavid du Colombier 	Bprint(&stdout, "%6ud", x);
358219b2ee8SDavid du Colombier 
359219b2ee8SDavid du Colombier 	x = vga->crt[0x06];
360219b2ee8SDavid du Colombier 	if(vga->crt[0x07] & 0x01)
361219b2ee8SDavid du Colombier 		x |= 0x100;
362219b2ee8SDavid du Colombier 	if(vga->crt[0x07] & 0x20)
363219b2ee8SDavid du Colombier 		x |= 0x200;
364219b2ee8SDavid du Colombier 	if(vga->crt[0x5E] & 0x01)
365219b2ee8SDavid du Colombier 		x |= 0x400;
366219b2ee8SDavid du Colombier 	x += 2;
367219b2ee8SDavid du Colombier 	if(interlace)
368219b2ee8SDavid du Colombier 		x *= 2;
369219b2ee8SDavid du Colombier 	printitem(name, "vt");
370219b2ee8SDavid du Colombier 	printreg(x);
3717dd7cddfSDavid du Colombier 	Bprint(&stdout, "%6ud", x);
372219b2ee8SDavid du Colombier }
373219b2ee8SDavid du Colombier 
374219b2ee8SDavid du Colombier Ctlr s3generic = {
375219b2ee8SDavid du Colombier 	"s3",				/* name */
376219b2ee8SDavid du Colombier 	snarf,				/* snarf */
377219b2ee8SDavid du Colombier 	0,				/* options */
378219b2ee8SDavid du Colombier 	init,				/* init */
379219b2ee8SDavid du Colombier 	load,				/* load */
380219b2ee8SDavid du Colombier 	dump,				/* dump */
381219b2ee8SDavid du Colombier };
382