VirtualBox

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

Last change on this file since 71963 was 69500, checked in by vboxsync, 7 years ago

*: scm --update-copyright-year

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 14.2 KB
Line 
1/* $Id: logo.c 69500 2017-10-28 15:14:05Z vboxsync $ */
2/** @file
3 * Stuff for drawing the BIOS logo.
4 */
5
6/*
7 * Copyright (C) 2004-2017 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 * @param mode 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 * @param mode 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 * @param mode 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 else if (!f12_pressed)
413 outw(LOGO_IO_PORT, LOGO_CMD_SHOW_BMP | 0);
414
415done:
416 // Clear forced boot drive setting.
417 write_byte(ebda_seg, (uint16_t)&EbdaData->uForceBootDevice, 0);
418
419 // Don't restore previous video mode
420 // The default text mode should be set up. (defect @bugref{1235})
421 set_mode(0x0003);
422
423 // If Setup menu enabled
424 if (uBootMenu)
425 {
426 // If the graphics logo disabled
427 if (!is_fade_in && !is_fade_out && !logo_time)
428 {
429 if (uBootMenu == 2)
430 printf("Press F12 to select boot device.\n");
431
432 // if the user has pressed F12 don't wait here
433 if (!f12_pressed)
434 {
435 // Wait for timeout or keystroke
436 scode = wait(F12_WAIT_TIME, 1);
437 if (scode == F12_SCAN_CODE)
438 f12_pressed = 1;
439 }
440 }
441
442 // If F12 pressed, show boot menu
443 if (f12_pressed)
444 {
445 uint8_t boot_device = 0;
446 uint8_t boot_drive = 0;
447
448 clear_screen();
449
450 // Show menu. Note that some versions of bcc freak out if we split these strings.
451 printf("\nVirtualBox temporary boot device selection\n\nDetected Hard disks:\n\n");
452 print_detected_harddisks();
453 printf("\nOther boot devices:\n f) Floppy\n c) CD-ROM\n l) LAN\n\n b) Continue booting\n");
454
455
456
457 // Wait for keystroke
458 for (;;)
459 {
460 do
461 {
462 scode = wait(WAIT_HZ, 1);
463 } while (scode == 0);
464
465 if (scode == 0x30)
466 {
467 // 'b' ... continue
468 break;
469 }
470
471 // Check if hard disk was selected
472 if ((scode >= 0x02) && (scode <= 0x09))
473 {
474 boot_drive = get_boot_drive(scode);
475
476 /*
477 * 0xff indicates that there is no mapping
478 * from the scan code to a hard drive.
479 * Wait for next keystroke.
480 */
481 if (boot_drive == 0xff)
482 continue;
483
484 write_byte(ebda_seg, (uint16_t)&EbdaData->uForceBootDrive, boot_drive);
485 boot_device = 0x02;
486 break;
487 }
488
489 switch (scode)
490 {
491 case 0x21:
492 // Floppy
493 boot_device = 0x01;
494 break;
495 case 0x2e:
496 // CD-ROM
497 boot_device = 0x03;
498 break;
499 case 0x26:
500 // LAN
501 boot_device = 0x04;
502 break;
503 }
504
505 if (boot_device != 0)
506 break;
507 }
508
509 write_byte(ebda_seg, (uint16_t)&EbdaData->uForceBootDevice, boot_device);
510
511 // Switch to text mode. Clears screen and enables cursor again.
512 set_mode(0x0003);
513 }
514 }
515
516 // Restore PIT ticks
517 wait_uninit();
518
519 return;
520}
521
522
523void delay_boot(uint16_t secs)
524{
525 uint16_t i;
526
527 if (!secs)
528 return;
529
530 // Set PIT to 1ms ticks
531 wait_init();
532
533 printf("Delaying boot for %d seconds:", secs);
534 for (i = secs; i > 0; i--)
535 {
536 printf(" %d", i);
537 wait(WAIT_HZ, 0);
538 }
539 printf("\n");
540 // Restore PIT ticks
541 wait_uninit();
542}
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