VirtualBox

source: vbox/trunk/src/VBox/Devices/PC/BIOS-new/logo.c@ 41658

Last change on this file since 41658 was 39375, checked in by vboxsync, 13 years ago

Straightened out BIOS AHCI code, reduced stack usage.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 13.4 KB
Line 
1/* $Id: logo.c 39375 2011-11-18 20:19:01Z vboxsync $ */
2/** @file
3 * Stuff for drawing the BIOS logo.
4 */
5
6/*
7 * Copyright (C) 2004-2011 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18#include <stdint.h>
19#include "biosint.h"
20#include "inlines.h"
21#include "ebda.h"
22
23#define WAIT_HZ 64
24#define WAIT_MS 16
25
26#define F12_SCAN_CODE 0x86
27#define F12_WAIT_TIME (3 * WAIT_HZ) /* 3 seconds. Used only if logo disabled. */
28
29#include <VBox/bioslogo.h>
30
31/**
32 * Set video mode (VGA).
33 * @params New video mode.
34 */
35void set_mode(uint8_t mode);
36#pragma aux set_mode = \
37 "mov ah, 0" \
38 "int 10h" \
39 parm [al] modify [ax] nomemory;
40
41
42/**
43 * Set VESA video mode.
44 * @params New video mode.
45 */
46uint16_t vesa_set_mode(uint16_t mode);
47#pragma aux vesa_set_mode = \
48 "mov ax, 4F02h" \
49 "int 10h" \
50 parm [bx] modify [ax] nomemory;
51
52
53/**
54 * Check for keystroke.
55 * @returns True if keystroke available, False if not.
56 */
57//@todo: INT 16h should already be returning the right value in al; could also use setz
58uint8_t check_for_keystroke(void);
59#pragma aux check_for_keystroke = \
60 "mov ax, 100h" \
61 "int 16h" \
62 "jz no_key" \
63 "mov al, 1" \
64 "jmp done" \
65 "no_key:" \
66 "xor al, al" \
67 "done:" \
68 modify [ax] nomemory;
69
70
71/**
72 * Get keystroke.
73 * @returns BIOS scan code.
74 */
75uint8_t get_keystroke(void);
76#pragma aux get_keystroke = \
77 "xor ax, ax" \
78 "int 16h" \
79 "xchg ah, al" \
80 modify [ax] nomemory;
81
82
83//@todo: This whole business with reprogramming the PIT is rather suspect.
84// The BIOS already has waiting facilities in INT 15h (fn 83h, 86h) which
85// should be utilized instead.
86
87// Set the timer to 16ms ticks (64K / (Hz / (PIT_HZ / 64K)) = count).
88void wait_init(void);
89#pragma aux wait_init = \
90 "mov al, 34h" \
91 "out 43h, al" \
92 "mov al, 0D3h" \
93 "out 40h, al" \
94 "mov al, 048h" \
95 "out 40h, al" \
96 modify [ax] nomemory;
97
98//@todo: using this private interface is not great
99extern void rtc_post(void);
100#pragma aux rtc_post "*";
101
102/* Restore the timer to the default 18.2Hz. Reinitialize the tick
103 * and rollover counts since we've screwed them up by running the
104 * timer at WAIT_HZ for a while.
105 */
106void wait_uninit(void);
107#pragma aux wait_uninit = \
108 ".386" \
109 "mov al, 34h" \
110 "out 43h, al" \
111 "xor ax, ax" \
112 "out 40h, al" \
113 "out 40h, al" \
114 "pushad" \
115 "push ds" \
116 "mov ds, ax" \
117 "call rtc_post" \
118 "pop ds" \
119 "popad" \
120 modify [ax] nomemory;
121
122
123/**
124 * Waits (sleeps) for the given number of ticks.
125 * Checks for keystroke.
126 *
127 * @returns BIOS scan code if available, 0 if not.
128 * @param ticks Number of ticks to sleep.
129 * @param stop_on_key Whether to stop immediately upon keypress.
130 */
131uint8_t wait(uint16_t ticks, uint8_t stop_on_key)
132{
133 long ticks_to_wait, delta;
134 uint16_t old_flags;
135 uint32_t prev_ticks, t;
136 uint8_t scan_code = 0;
137
138 /*
139 * We may or may not be called with interrupts disabled. For the duration
140 * of this function, interrupts must be enabled.
141 */
142 old_flags = int_query();
143 int_enable();
144
145 /*
146 * The 0:046c wraps around at 'midnight' according to a 18.2Hz clock.
147 * We also have to be careful about interrupt storms.
148 */
149 ticks_to_wait = ticks;
150 prev_ticks = read_dword(0x0, 0x46c);
151 do
152 {
153 halt();
154 t = read_dword(0x0, 0x46c);
155 if (t > prev_ticks)
156 {
157 delta = t - prev_ticks; /* The temp var is required or bcc screws up. */
158 ticks_to_wait -= delta;
159 }
160 else if (t < prev_ticks)
161 ticks_to_wait -= t; /* wrapped */
162 prev_ticks = t;
163
164 if (check_for_keystroke())
165 {
166 scan_code = get_keystroke();
167 bios_printf(BIOS_PRINTF_INFO, "Key pressed: %x\n", scan_code);
168 if (stop_on_key)
169 return scan_code;
170 }
171 } while (ticks_to_wait > 0);
172 int_restore(old_flags);
173 return scan_code;
174}
175
176uint8_t read_logo_byte(uint8_t offset)
177{
178 outw(LOGO_IO_PORT, LOGO_CMD_SET_OFFSET | offset);
179 return inb(LOGO_IO_PORT);
180}
181
182uint16_t read_logo_word(uint8_t offset)
183{
184 outw(LOGO_IO_PORT, LOGO_CMD_SET_OFFSET | offset);
185 return inw(LOGO_IO_PORT);
186}
187
188// Hide cursor, clear screen and move cursor to starting position
189void clear_screen(void);
190#pragma aux clear_screen = \
191 "mov ax, 100h" \
192 "mov cx, 1000h" \
193 "int 10h" \
194 "mov ax, 700h" \
195 "mov bh, 7" \
196 "xor cx, cx" \
197 "mov dx, 184Fh" \
198 "int 10h" \
199 "mov ax, 200h" \
200 "xor bx, bx" \
201 "xor dx, dx" \
202 "int 10h" \
203 modify [ax bx cx dx] nomemory;
204
205void print_detected_harddisks(void)
206{
207 uint16_t ebda_seg=read_word(0x0040,0x000E);
208 uint8_t hd_count;
209 uint8_t hd_curr = 0;
210 uint8_t ide_ctrl_printed = 0;
211 uint8_t sata_ctrl_printed = 0;
212 uint8_t scsi_ctrl_printed = 0;
213 uint8_t device;
214
215 hd_count = read_byte(ebda_seg, (uint16_t)&EbdaData->bdisk.hdcount);
216
217 for (hd_curr = 0; hd_curr < hd_count; hd_curr++)
218 {
219 device = read_byte(ebda_seg, (uint16_t)&EbdaData->bdisk.hdidmap[hd_curr]);
220
221#ifdef VBOX_WITH_AHCI
222 if (VBOX_IS_AHCI_DEVICE(device))
223 {
224 if (sata_ctrl_printed == 0)
225 {
226 printf("\n\nAHCI controller:\n");
227 sata_ctrl_printed = 1;
228 }
229
230 printf("\n %d) Hard disk", hd_curr+1);
231
232 }
233 else
234#endif
235#ifdef VBOX_WITH_SCSI
236 if (VBOX_IS_SCSI_DEVICE(device))
237 {
238 if (scsi_ctrl_printed == 0)
239 {
240 printf("\n\nSCSI controller:\n");
241 scsi_ctrl_printed = 1;
242 }
243
244 printf("\n %d) Hard disk", hd_curr+1);
245
246 }
247 else
248#endif
249 {
250
251 if ((device < 4) && (ide_ctrl_printed == 0))
252 {
253 printf("IDE controller:\n");
254 ide_ctrl_printed = 1;
255 }
256 else if ((device >= 4) && (sata_ctrl_printed == 0))
257 {
258 printf("\n\nAHCI controller:\n");
259 sata_ctrl_printed = 1;
260 }
261
262 printf("\n %d) ", hd_curr+1);
263
264 /*
265 * If actual_device is bigger than or equal 4
266 * this is the next controller and
267 * the positions start at the beginning.
268 */
269 if (device >= 4)
270 device -= 4;
271
272 if (device / 2)
273 printf("Secondary ");
274 else
275 printf("Primary ");
276
277 if (device % 2)
278 printf("Slave");
279 else
280 printf("Master");
281 }
282 }
283
284 if ( (ide_ctrl_printed == 0)
285 && (sata_ctrl_printed == 0)
286 && (scsi_ctrl_printed == 0))
287 printf("No hard disks found");
288
289 printf("\n");
290}
291
292uint8_t get_boot_drive(uint8_t scode)
293{
294 uint16_t ebda_seg=read_word(0x0040,0x000E);
295
296 /* Check that the scan code is in the range of detected hard disks. */
297 uint8_t hd_count = read_byte(ebda_seg, (uint16_t)&EbdaData->bdisk.hdcount);
298
299 /* The key '1' has scancode 0x02 which represents the first disk */
300 scode -= 2;
301
302 if (scode < hd_count)
303 return scode;
304
305 /* Scancode is higher than number of available devices */
306 return 0xff;
307}
308
309void show_logo(void)
310{
311 uint16_t ebda_seg = read_word(0x0040,0x000E);
312 uint8_t f12_pressed = 0;
313 uint8_t scode;
314 uint16_t tmp, i;
315
316 LOGOHDR *logo_hdr = 0;
317 uint8_t is_fade_in, is_fade_out, uBootMenu;
318 uint16_t logo_time;
319
320
321 // Set PIT to 1ms ticks
322 wait_init();
323
324
325 // Get main signature
326 tmp = read_logo_word((uint8_t)&logo_hdr->u16Signature);
327 if (tmp != 0x66BB)
328 goto done;
329
330 // Get options
331 is_fade_in = read_logo_byte((uint8_t)&logo_hdr->fu8FadeIn);
332 is_fade_out = read_logo_byte((uint8_t)&logo_hdr->fu8FadeOut);
333 logo_time = read_logo_word((uint8_t)&logo_hdr->u16LogoMillies);
334 uBootMenu = read_logo_byte((uint8_t)&logo_hdr->fu8ShowBootMenu);
335
336 // Is Logo disabled?
337 if (!is_fade_in && !is_fade_out && !logo_time)
338 goto done;
339
340 // Set video mode #0x142 640x480x32bpp
341 vesa_set_mode(0x142);
342
343 if (is_fade_in)
344 {
345 for (i = 0; i <= LOGO_SHOW_STEPS; i++)
346 {
347 outw(LOGO_IO_PORT, LOGO_CMD_SHOW_BMP | i);
348 scode = wait(16 / WAIT_MS, 0);
349 if (scode == F12_SCAN_CODE)
350 {
351 f12_pressed = 1;
352 break;
353 }
354 }
355 }
356 else
357 outw(LOGO_IO_PORT, LOGO_CMD_SHOW_BMP | LOGO_SHOW_STEPS);
358
359 // Wait (interval in milliseconds)
360 if (!f12_pressed)
361 {
362 scode = wait(logo_time / WAIT_MS, 1);
363 if (scode == F12_SCAN_CODE)
364 f12_pressed = 1;
365 }
366
367 // Fade out (only if F12 was not pressed)
368 if (is_fade_out && !f12_pressed)
369 {
370 for (i = LOGO_SHOW_STEPS; i > 0 ; i--)
371 {
372 outw(LOGO_IO_PORT, LOGO_CMD_SHOW_BMP | i);
373 scode = wait(16 / WAIT_MS, 0);
374 if (scode == F12_SCAN_CODE)
375 {
376 f12_pressed = 1;
377 break;
378 }
379 }
380 }
381
382done:
383 // Clear forced boot drive setting.
384 write_byte(ebda_seg, (uint16_t)&EbdaData->uForceBootDevice, 0);
385
386 // Don't restore previous video mode
387 // The default text mode should be set up. (defect #1235)
388 set_mode(0x0003);
389
390 // If Setup menu enabled
391 if (uBootMenu)
392 {
393 // If the graphics logo disabled
394 if (!is_fade_in && !is_fade_out && !logo_time)
395 {
396 if (uBootMenu == 2)
397 printf("Press F12 to select boot device.\n");
398
399 // if the user has pressed F12 don't wait here
400 if (!f12_pressed)
401 {
402 // Wait for timeout or keystroke
403 scode = wait(F12_WAIT_TIME, 1);
404 if (scode == F12_SCAN_CODE)
405 f12_pressed = 1;
406 }
407 }
408
409 // If F12 pressed, show boot menu
410 if (f12_pressed)
411 {
412 uint8_t boot_device = 0;
413 uint8_t boot_drive = 0;
414
415 clear_screen();
416
417 // Show menu. Note that some versions of bcc freak out if we split these strings.
418 printf("\nVirtualBox temporary boot device selection\n\nDetected Hard disks:\n\n");
419 print_detected_harddisks();
420 printf("\nOther boot devices:\n f) Floppy\n c) CD-ROM\n l) LAN\n\n b) Continue booting\n");
421
422
423
424 // Wait for keystroke
425 for (;;)
426 {
427 do
428 {
429 scode = wait(WAIT_HZ, 1);
430 } while (scode == 0);
431
432 if (scode == 0x30)
433 {
434 // 'b' ... continue
435 break;
436 }
437
438 // Check if hard disk was selected
439 if ((scode >= 0x02) && (scode <= 0x09))
440 {
441 boot_drive = get_boot_drive(scode);
442
443 /*
444 * 0xff indicates that there is no mapping
445 * from the scan code to a hard drive.
446 * Wait for next keystroke.
447 */
448 if (boot_drive == 0xff)
449 continue;
450
451 write_byte(ebda_seg, (uint16_t)&EbdaData->uForceBootDrive, boot_drive);
452 boot_device = 0x02;
453 break;
454 }
455
456 switch (scode)
457 {
458 case 0x21:
459 // Floppy
460 boot_device = 0x01;
461 break;
462 case 0x2e:
463 // CD-ROM
464 boot_device = 0x03;
465 break;
466 case 0x26:
467 // LAN
468 boot_device = 0x04;
469 break;
470 }
471
472 if (boot_device != 0)
473 break;
474 }
475
476 write_byte(ebda_seg, (uint16_t)&EbdaData->uForceBootDevice, boot_device);
477
478 // Switch to text mode. Clears screen and enables cursor again.
479 set_mode(0x0003);
480 }
481 }
482
483 // Restore PIT ticks
484 wait_uninit();
485
486 return;
487}
488
489
490void delay_boot(uint16_t secs)
491{
492 uint16_t i;
493
494 if (!secs)
495 return;
496
497 // Set PIT to 1ms ticks
498 wait_init();
499
500 printf("Delaying boot for %d seconds:", secs);
501 for (i = secs; i > 0; i--)
502 {
503 printf(" %d", i);
504 wait(WAIT_HZ, 0);
505 }
506 printf("\n");
507 // Restore PIT ticks
508 wait_uninit();
509}
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette