VirtualBox

source: vbox/trunk/src/VBox/Devices/PC/BIOS/logo.c@ 63547

Last change on this file since 63547 was 62509, checked in by vboxsync, 8 years ago

(C) 2016

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