2 Copyright © 2007-2012, The AROS Development Team. All rights reserved.
5 Desc: Real-mode code to set VBE mode.
9 #define _IMPLEMENTATION_
14 #define ABS(x) (((x) >= 0) ? (x) : -(x))
16 asm (".long getControllerInfo");
17 asm (".long getModeInfo");
18 asm (".long findMode");
19 asm (".long setVbeMode");
20 asm (".long paletteWidth");
21 asm (".long controllerinfo");
22 asm (".long modeinfo");
23 asm (".long timings");
25 short getControllerInfo(void)
29 controllerinfo
.signature
[0] = 'V';
30 controllerinfo
.signature
[1] = 'B';
31 controllerinfo
.signature
[2] = 'E';
32 controllerinfo
.signature
[3] = '2';
33 asm volatile("call go16 \n\t.code16 \n\t"
34 "movw $0x4f00, %%ax\n\t"
37 "DATA32 call go32\n\t.code32\n\t":"=b"(retval
):"D"(&controllerinfo
):"eax","ecx","cc");
41 /* In VBE 1.1 information about standard modes was optional,
42 so we use a hardcoded table here (we rely on this information) */
43 struct vesa11Info vesa11Modes
[] = {
54 short getModeInfo(long mode
)
58 char *ptr
= (char *)&modeinfo
;
59 for (i
= 0; i
< sizeof(modeinfo
); i
++)
61 asm volatile("call go16 \n\t.code16 \n\t"
62 "movw $0x4f01, %%ax\n\t"
65 "DATA32 call go32\n\t.code32\n\t":"=b"(retval
):"c"(mode
),"D"(&modeinfo
):"eax","cc");
66 if ((controllerinfo
.version
< 0x0102) && (mode
> 0x0FF) && (mode
< 0x108)) {
68 modeinfo
.x_resolution
= vesa11Modes
[i
].x_resolution
;
69 modeinfo
.y_resolution
= vesa11Modes
[i
].y_resolution
;
70 modeinfo
.bits_per_pixel
= vesa11Modes
[i
].bits_per_pixel
;
71 modeinfo
.memory_model
= vesa11Modes
[i
].memory_model
;
76 short setVbeMode(long mode
, BOOL set_refresh
)
80 /* Enable custom timings if possible */
81 if (set_refresh
&& controllerinfo
.version
>= 0x0300)
86 asm volatile("call go16 \n\t.code16 \n\t"
87 "movw $0x4f02, %%ax\n\t"
90 "DATA32 call go32\n\t.code32\n\t":"=b"(retval
):"0"(mode
),"D"(&timings
):"eax","ecx","cc");
94 short paletteWidth(long req
, unsigned char* width
)
97 unsigned char reswidth
;
99 asm volatile("call go16\n\t.code16\n\t"
100 "movw $0x4f08, %%ax\n\t"
104 "DATA32 call go32\n\t.code32\n\t":"=b"(retval
),"=c"(reswidth
):"0"(req
):"eax","cc");
109 /* Definitions used in CVT formula */
114 #define DUTY_CYCLE(period) \
115 (((C - J) / 2 + J) * 1000 - (M / 2 * (period) / 1000))
116 #define MIN_DUTY_CYCLE 20 /* % */
117 #define MIN_V_PORCH 3 /* lines */
118 #define MIN_V_PORCH_TIME 550 /* us */
119 #define CLOCK_STEP 250000 /* Hz */
121 /* Partial implementation of CVT formula */
122 void calcTimings(int vfreq
)
124 ULONG x
, y
, h_period
, h_freq
, h_total
, h_blank
, h_front
, h_sync
, h_back
,
125 v_freq
, v_total
, v_front
, v_sync
, v_back
, duty_cycle
, pixel_freq
;
127 x
= modeinfo
.x_resolution
;
128 y
= modeinfo
.y_resolution
;
130 /* Get horizontal period in microseconds */
131 h_period
= (1000000000 / vfreq
- MIN_V_PORCH_TIME
* 1000)
134 /* Vertical front porch is fixed */
135 v_front
= MIN_V_PORCH
;
137 /* Use aspect ratio to determine V-sync lines */
140 else if (x
== y
* 16 / 9)
142 else if (x
== y
* 16 / 10)
144 else if (x
== y
* 5 / 4)
146 else if (x
== y
* 15 / 9)
151 /* Get vertical back porch */
152 v_back
= MIN_V_PORCH_TIME
* 1000 / h_period
+ 1;
153 if (v_back
< MIN_V_PORCH
)
154 v_back
= MIN_V_PORCH
;
157 /* Get total lines per frame */
158 v_total
= y
+ v_front
+ v_sync
+ v_back
;
160 /* Get horizontal blanking pixels */
161 duty_cycle
= DUTY_CYCLE(h_period
);
162 if (duty_cycle
< MIN_DUTY_CYCLE
)
163 duty_cycle
= MIN_DUTY_CYCLE
;
165 h_blank
= 10 * x
* duty_cycle
/ (100000 - duty_cycle
);
166 h_blank
/= 2 * 8 * 10;
167 h_blank
= h_blank
* (2 * 8);
169 /* Get total pixels in a line */
170 h_total
= x
+ h_blank
;
172 /* Calculate frequencies for each pixel, line and field */
173 h_freq
= 1000000000 / h_period
;
174 pixel_freq
= h_freq
* h_total
/ CLOCK_STEP
* CLOCK_STEP
;
175 h_freq
= pixel_freq
/ h_total
;
176 v_freq
= 100 * h_freq
/ v_total
;
178 /* Back porch is half of H-blank */
179 h_back
= h_blank
/ 2;
181 /* H-sync is a fixed percentage of H-total */
182 h_sync
= h_total
/ 100 * 8;
184 /* Front porch is whatever's left */
185 h_front
= h_blank
- h_sync
- h_back
;
187 /* Fill in VBE timings structure */
188 timings
.h_total
= h_total
;
189 timings
.h_sync_start
= x
+ h_front
;
190 timings
.h_sync_end
= h_total
- h_back
;
191 timings
.v_total
= v_total
;
192 timings
.v_sync_start
= y
+ v_front
;
193 timings
.v_sync_end
= v_total
- v_back
;
195 timings
.pixel_clock
= pixel_freq
;
196 timings
.refresh_rate
= v_freq
;
199 short findMode(int x
, int y
, int d
, int vfreq
, BOOL prioritise_depth
)
201 unsigned long match
, bestmatch
= 0, matchd
, bestmatchd
= 0;
202 unsigned short bestmode
= 0xffff, mode_attrs
;
205 if (getControllerInfo() == 0x4f)
207 unsigned short *modes
= (unsigned short *)
208 (((controllerinfo
.video_mode
& 0xffff0000) >> 12) + (controllerinfo
.video_mode
& 0xffff));
212 if (controllerinfo
.version
< 0x0200)
217 for (i
=0; modes
[i
] != 0xffff; ++i
)
219 if (getModeInfo(modes
[i
])!= 0x4f) continue;
220 if ((modeinfo
.mode_attributes
& mode_attrs
) != mode_attrs
) continue;
221 if ((modeinfo
.memory_model
!= 6) && (modeinfo
.memory_model
!= 4))
223 if ((modeinfo
.memory_model
== 4) && (modeinfo
.mode_attributes
& 0x20))
226 /* Return immediately if an exactly matching mode is found
227 * (otherwise we could potentially return a mode with the right
228 * area but different dimensions) */
229 if (modeinfo
.x_resolution
== x
&&
230 modeinfo
.y_resolution
== y
&&
231 modeinfo
.bits_per_pixel
== d
)
237 match
= ABS(modeinfo
.x_resolution
*modeinfo
.y_resolution
- x
*y
);
238 matchd
= modeinfo
.bits_per_pixel
>= d
? modeinfo
.bits_per_pixel
-d
: (d
-modeinfo
.bits_per_pixel
)*2;
240 if (prioritise_depth
)
242 /* Check if current mode is the best so far at the desired
243 * depth, or has a higher depth than previously found */
244 if (bestmode
== 0xffff || (match
< bestmatch
245 && modeinfo
.bits_per_pixel
== bestd
)
246 || (bestd
< d
&& modeinfo
.bits_per_pixel
> bestd
247 && modeinfo
.bits_per_pixel
<= d
))
251 bestd
= modeinfo
.bits_per_pixel
;
256 /* Check if current mode either has the closest resolution
257 * so far to that requested, or is equally close as the
258 * previous best but has closer colour depth */
259 if (bestmode
== 0xffff || match
< bestmatch
260 || (match
== bestmatch
&& matchd
< bestmatchd
))
270 /* Set up timings to achieve the desired refresh rate */
271 if (controllerinfo
.version
>= 0x0300 && getModeInfo(bestmode
) == 0x4f)
278 " .code16\n\t.globl go32\n\t.type go32,@function\n"
279 "go32: DATA32 ADDR32 lgdt GDT_reg\n"
285 "1: movw $0x10, %ax\n"
291 " movl (%esp), %ecx\n"
292 " movl stack32, %eax\n"
294 " movl %ecx, (%esp)\n"
300 " .code32\n\t.globl go16\n\t.type go16,@function\n"
301 "go16: lgdt GDT_reg\n"
304 " movl %esp, stack32\n"
305 " movl (%esp), %eax\n"
306 " movl %eax, scratch + 63\n"
307 " movl $scratch + 63, %esp\n"
314 " ljmp $0x18, $1f\n\t.code16\n"
319 " DATA32 ljmp $0x00, $1f\n"
330 const unsigned long long GDT_Table
[] = {
331 0x0000000000000000ULL
,
332 0x00cf9a000000ffffULL
, /* Code32 */
333 0x00cf92000000ffffULL
, /* Data32 */
334 0x00009e000000ffffULL
, /* Code16 */
335 0x000092000000ffffULL
/* Data16 */
340 unsigned short l1
__attribute__((packed
));
341 const void *l3
__attribute__((packed
));
343 GDT_reg
= {sizeof(GDT_Table
)-1, GDT_Table
},
344 IDT_reg16
= {0x400, 0},
347 unsigned long stack32
;
348 unsigned long scratch
[64];
349 struct vbe_controller controllerinfo
;
350 struct vbe_mode modeinfo
;
351 struct CRTCInfoBlock timings
;