9 * Generic S3 GUI Accelerator.
12 snarf(Vga* vga, Ctlr* ctlr)
16 trace("%s->snarf->s3generic\n", ctlr->name);
19 * Unlock extended registers.
20 * 0xA5 ensures Crt36 and Crt37 are also unlocked
21 * (0xA0 unlocks everything else).
23 vgaxo(Crtx, 0x38, 0x48);
24 vgaxo(Crtx, 0x39, 0xA5);
27 * Not all registers exist on all chips.
28 * Crt3[EF] don't exist on any.
30 for(i = 0x30; i < 0x70; i++)
31 vga->crt[i] = vgaxi(Crtx, i);
36 switch((vga->crt[0x36]>>5) & 0x07){
39 vga->vmz = 4*1024*1024;
43 vga->vmz = 3*1024*1024;
47 vga->vmz = 2*1024*1024;
51 vga->vmz = 1*1024*1024;
63 init(Vga* vga, Ctlr* ctlr)
68 trace("%s->init->s3generic\n", ctlr->name);
72 * Is enhanced mode is necessary?
74 if((ctlr->flag & (Uenhanced|Henhanced)) == Henhanced){
76 resyncinit(vga, ctlr, Uenhanced, 0);
78 resyncinit(vga, ctlr, 0, Uenhanced|Henhanced);
80 if((ctlr->flag & Uenhanced) == 0 && mode->x > 1024)
81 error("%s: no support for 1-bit mode > 1024x768x1\n", ctlr->name);
83 vga->crt[0x31] = 0x85;
84 vga->crt[0x6A] &= 0xC0;
85 vga->crt[0x32] &= ~0x40;
87 vga->crt[0x31] |= 0x08;
88 vga->crt[0x32] |= 0x40;
90 vga->crt[0x33] |= 0x20;
92 vga->crt[0x3A] |= 0x10;
94 vga->crt[0x3A] &= ~0x10;
96 vga->crt[0x34] = 0x10;
97 vga->crt[0x35] = 0x00;
99 vga->crt[0x3C] = vga->crt[0]/2;
100 vga->crt[0x42] |= 0x20;
103 vga->crt[0x3C] = 0x00;
104 vga->crt[0x42] &= ~0x20;
107 vga->crt[0x40] = (vga->crt[0x40] & 0xF2);
108 vga->crt[0x43] = 0x00;
109 vga->crt[0x45] = 0x00;
111 vga->crt[0x50] &= 0x3E;
114 else if(mode->x <= 800)
116 else if(mode->x <= 1024)
118 else if(mode->x <= 1152)
120 else if(mode->x <= 1280)
126 vga->crt[0x51] = (vga->crt[0x51] & 0xC3)|((vga->crt[0x13]>>4) & 0x30);
127 vga->crt[0x53] &= ~0x10;
130 * Set up linear aperture. For the moment it's 64K at 0xA0000.
131 * The real base address will be assigned before load is called.
133 vga->crt[0x58] = 0x88;
134 if(ctlr->flag & Uenhanced){
135 vga->crt[0x58] |= 0x10;
136 if(vga->linear && (ctlr->flag & Hlinear))
137 ctlr->flag |= Ulinear;
138 if(vga->vmz <= 1024*1024)
139 vga->vma = 1024*1024;
140 else if(vga->vmz <= 2*1024*1024)
141 vga->vma = 2*1024*1024;
143 vga->vma = 8*1024*1024;
145 vga->crt[0x59] = 0x00;
146 vga->crt[0x5A] = 0x0A;
148 vga->crt[0x5D] &= 0x80;
149 if(vga->crt[0x00] & 0x100)
150 vga->crt[0x5D] |= 0x01;
151 if(vga->crt[0x01] & 0x100)
152 vga->crt[0x5D] |= 0x02;
153 if(vga->crt[0x02] & 0x100)
154 vga->crt[0x5D] |= 0x04;
155 if(vga->crt[0x04] & 0x100)
156 vga->crt[0x5D] |= 0x10;
157 if(vga->crt[0x3B] & 0x100)
158 vga->crt[0x5D] |= 0x40;
160 vga->crt[0x5E] = 0x40;
161 if(vga->crt[0x06] & 0x400)
162 vga->crt[0x5E] |= 0x01;
163 if(vga->crt[0x12] & 0x400)
164 vga->crt[0x5E] |= 0x02;
165 if(vga->crt[0x15] & 0x400)
166 vga->crt[0x5E] |= 0x04;
167 if(vga->crt[0x10] & 0x400)
168 vga->crt[0x5E] |= 0x10;
170 ctlr->type = s3generic.name;
176 load(Vga* vga, Ctlr* ctlr)
180 trace("%s->load->s3generic\n", ctlr->name);
182 vgaxo(Crtx, 0x31, vga->crt[0x31]);
183 vgaxo(Crtx, 0x32, vga->crt[0x32]);
184 vgaxo(Crtx, 0x33, vga->crt[0x33]);
185 vgaxo(Crtx, 0x34, vga->crt[0x34]);
186 vgaxo(Crtx, 0x35, vga->crt[0x35]);
187 vgaxo(Crtx, 0x3A, vga->crt[0x3A]);
188 vgaxo(Crtx, 0x3B, vga->crt[0x3B]);
189 vgaxo(Crtx, 0x3C, vga->crt[0x3C]);
191 vgaxo(Crtx, 0x40, vga->crt[0x40]|0x01);
192 vgaxo(Crtx, 0x42, vga->crt[0x42]);
193 vgaxo(Crtx, 0x43, vga->crt[0x43]);
194 vgaxo(Crtx, 0x45, vga->crt[0x45]);
196 vgaxo(Crtx, 0x50, vga->crt[0x50]);
197 vgaxo(Crtx, 0x51, vga->crt[0x51]);
198 vgaxo(Crtx, 0x53, vga->crt[0x53]);
199 vgaxo(Crtx, 0x54, vga->crt[0x54]);
200 vgaxo(Crtx, 0x55, vga->crt[0x55]);
202 if(ctlr->flag & Ulinear){
204 vga->crt[0x59] = (l>>8) & 0xFF;
205 vga->crt[0x5A] = l & 0xFF;
206 if(vga->vmz <= 1024*1024)
207 vga->crt[0x58] |= 0x01;
208 else if(vga->vmz <= 2*1024*1024)
209 vga->crt[0x58] |= 0x02;
211 vga->crt[0x58] |= 0x03;
213 vgaxo(Crtx, 0x59, vga->crt[0x59]);
214 vgaxo(Crtx, 0x5A, vga->crt[0x5A]);
215 vgaxo(Crtx, 0x58, vga->crt[0x58]);
217 vgaxo(Crtx, 0x5D, vga->crt[0x5D]);
218 vgaxo(Crtx, 0x5E, vga->crt[0x5E]);
220 vgaxo(Crtx, 0x6A, vga->crt[0x6A]);
226 dump(Vga* vga, Ctlr* ctlr)
228 int i, id, interlace, mul, div;
234 printitem(name, "Crt30");
235 for(i = 0x30; i < 0x3E; i++)
236 printreg(vga->crt[i]);
238 printitem(name, "Crt40");
239 for(i = 0x40; i < 0x50; i++)
240 printreg(vga->crt[i]);
242 printitem(name, "Crt50");
243 for(i = 0x50; i < 0x60; i++)
244 printreg(vga->crt[i]);
246 printitem(name, "Crt60");
247 for(i = 0x60; i < 0x70; i++)
248 printreg(vga->crt[i]);
251 * Try to disassemble the snarfed values into
252 * understandable numbers.
253 * Only do this if we weren't called after Finit.
255 if(ctlr->flag & Finit)
260 * If hde <= 400, assume this is a 928 or Vision964
261 * and the horizontal values have been divided by 4.
263 * if ViRGE/[DG]X and 15 or 16bpp, horizontal values have
264 * been multiplied by 2.
269 if(strcmp(name, "virge") == 0){
270 id = (vga->crt[0x2D]<<8)|vga->crt[0x2E];
272 if(id==0x8A01 && ((vga->crt[0x67] & 0x30) || (vga->crt[0x67] & 0x50))){
279 if(vga->crt[0x5D] & 0x02)
289 printitem(name, "hde");
291 Bprint(&stdout, "%6ud", x);
293 shb = vga->crt[0x02];
294 if(vga->crt[0x5D] & 0x04)
297 shb = (shb * mul) / div;
298 printitem(name, "shb");
300 Bprint(&stdout, "%6ud", shb);
302 x = vga->crt[0x03] & 0x1F;
303 if(vga->crt[0x05] & 0x80)
306 x = shb|x; /* ???? */
307 if(vga->crt[0x5D] & 0x08)
309 printitem(name, "ehb");
311 Bprint(&stdout, "%6ud", x);
314 if(vga->crt[0x5D] & 0x01)
318 printitem(name, "ht");
320 Bprint(&stdout, "%6ud", x);
322 interlace = vga->crt[0x42] & 0x20;
324 if(vga->crt[0x07] & 0x02)
326 if(vga->crt[0x07] & 0x40)
328 if(vga->crt[0x5E] & 0x02)
333 printitem(name, "vde");
335 Bprint(&stdout, "%6ud", x);
337 vrs = vga->crt[0x10];
338 if(vga->crt[0x07] & 0x04)
340 if(vga->crt[0x07] & 0x80)
342 if(vga->crt[0x5E] & 0x10)
346 printitem(name, "vrs");
348 Bprint(&stdout, "%6ud", vrs);
352 x = (vrs & ~0x0F)|(vga->crt[0x11] & 0x0F);
355 printitem(name, "vre");
357 Bprint(&stdout, "%6ud", x);
360 if(vga->crt[0x07] & 0x01)
362 if(vga->crt[0x07] & 0x20)
364 if(vga->crt[0x5E] & 0x01)
369 printitem(name, "vt");
371 Bprint(&stdout, "%6ud", x);