VirtualBox

source: vbox/trunk/src/VBox/Devices/PC/BIOS/orgs.asm@ 62495

Last change on this file since 62495 was 61054, checked in by vboxsync, 9 years ago

BIOS: Enable x2APIC mode when asked.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 42.1 KB
Line 
1;;
2;; Copyright (C) 2006-2015 Oracle Corporation
3;;
4;; This file is part of VirtualBox Open Source Edition (OSE), as
5;; available from http://www.virtualbox.org. This file is free software;
6;; you can redistribute it and/or modify it under the terms of the GNU
7;; General Public License (GPL) as published by the Free Software
8;; Foundation, in version 2 as it comes in the "COPYING" file of the
9;; VirtualBox OSE distribution. VirtualBox OSE is distributed in the
10;; hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
11;; --------------------------------------------------------------------
12;;
13;; This code is based on:
14;;
15;; ROM BIOS for use with Bochs/Plex86/QEMU emulation environment
16;;
17;; Copyright (C) 2002 MandrakeSoft S.A.
18;;
19;; MandrakeSoft S.A.
20;; 43, rue d'Aboukir
21;; 75002 Paris - France
22;; http://www.linux-mandrake.com/
23;; http://www.mandrakesoft.com/
24;;
25;; This library is free software; you can redistribute it and/or
26;; modify it under the terms of the GNU Lesser General Public
27;; License as published by the Free Software Foundation; either
28;; version 2 of the License, or (at your option) any later version.
29;;
30;; This library is distributed in the hope that it will be useful,
31;; but WITHOUT ANY WARRANTY; without even the implied warranty of
32;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
33;; Lesser General Public License for more details.
34;;
35;; You should have received a copy of the GNU Lesser General Public
36;; License along with this library; if not, write to the Free Software
37;; Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
38;;
39;;
40
41; Oracle LGPL Disclaimer: For the avoidance of doubt, except that if any license choice
42; other than GPL or LGPL is available it will apply instead, Oracle elects to use only
43; the Lesser General Public License version 2.1 (LGPLv2) at this time for any software where
44; a choice of LGPL license versions is made available with the language indicating
45; that LGPLv2 or any later version may be used, or where a choice of which version
46; of the LGPL is applied is otherwise unspecified.
47
48
49include commondefs.inc
50
51EBDA_SEG equ 09FC0h ; starts at 639K
52EBDA_SIZE equ 1 ; 1K
53BASE_MEM_IN_K equ (640 - EBDA_SIZE)
54
55CMOS_ADDR equ 070h
56CMOS_DATA equ 071h
57
58
59PIC_CMD_EOI equ 020h
60PIC_MASTER equ 020h
61PIC_SLAVE equ 0A0h
62
63BIOS_FIX_BASE equ 0E000h
64
65if VBOX_BIOS_CPU ge 80286
66SYS_MODEL_ID equ 0FCh ; PC/AT
67else
68SYS_MODEL_ID equ 0FBh ; PC/XT
69endif
70SYS_SUBMODEL_ID equ 0
71BIOS_REVISION equ 1
72
73BIOS_BUILD_DATE equ '06/23/99'
74BIOS_COPYRIGHT equ 'Oracle VM VirtualBox BIOS'
75
76BX_ROMBIOS32 equ 0
77BX_CALL_INT15_4F equ 1
78
79;; Set a fixed BIOS location, with a marker for verification
80BIOSORG macro addr, addr_minus_two
81.errnz (addr - 2 - addr_minus_two) ;; Couldn't convince wasm to accept $ here. Would've save us a lot of bother and ugly SED.
82 BIOSORG_CHECK_BEFORE addr_minus_two
83 org addr - BIOS_FIX_BASE - 2
84 db 'XM'
85 BIOSORG_CHECK addr
86 endm
87
88;; Set an interrupt vector (not very efficient if multiple vectors are
89;; programmed in one go)
90SET_INT_VECTOR macro vec, segm, offs
91 mov ax, offs
92 mov ds:[vec*4], ax
93 mov ax, segm
94 mov ds:[vec*4+2], ax
95endm
96
97; Set up an environment C code expects. DS must point to the BIOS segment
98; and the direction flag must be cleared(!)
99C_SETUP macro
100 push cs
101 pop ds
102 cld
103endm
104
105
106;; External function in separate modules
107extrn _dummy_isr_function:near
108extrn _log_bios_start:near
109extrn _nmi_handler_msg:near
110extrn _int18_panic_msg:near
111extrn _int09_function:near
112extrn _int13_diskette_function:near
113extrn _int13_eltorito:near
114extrn _int13_cdemu:near
115extrn _int13_cdrom:near
116extrn _cdemu_isactive:near
117extrn _cdemu_emulated_drive:near
118extrn _int13_harddisk:near
119extrn _int13_harddisk_ext:near
120extrn _int14_function:near
121extrn _int15_function:near
122extrn _int15_function_mouse:near
123extrn _int15_function32:near
124extrn _int16_function:near
125extrn _int17_function:near
126extrn _int19_function:near
127extrn _int1a_function:near
128extrn _pci16_function:near
129extrn _int70_function:near
130extrn _int74_function:near
131extrn _apm_function:near
132extrn _ata_init:near
133extrn _scsi_init:near
134extrn _ata_detect:near
135extrn _cdemu_init:near
136extrn _keyboard_init:near
137extrn _print_bios_banner:near
138extrn _inv_op_handler:near
139extrn rom_scan_:near
140ifdef VBOX_WITH_AHCI
141extrn _ahci_init:near
142endif
143if VBOX_BIOS_CPU ge 80286
144extrn _int15_blkmove:near
145endif
146if VBOX_BIOS_CPU ge 80386
147extrn _apic_setup:near
148endif
149
150
151;; Symbols referenced from C code
152public _diskette_param_table
153public _pmode_IDT
154public _rmode_IDT
155public post
156public eoi_both_pics
157public rtc_post
158
159;; Additional publics for easier disassembly and debugging
160ifndef DEBUG
161 DEBUG equ 1
162endif
163ifdef DEBUG
164
165public int08_handler
166public int0e_handler
167public int11_handler
168public int12_handler
169public int13_handler
170public int13_relocated
171if VBOX_BIOS_CPU eq 8086
172public jmp_call_ret_int13_out
173endif
174public int15_handler
175public int17_handler
176public int19_handler
177public int19_relocated
178public dummy_iret
179public nmi
180public rom_fdpt
181public cpu_reset
182public normal_post
183public eoi_jmp_post
184public no_eoi_jmp_post
185public eoi_master_pic
186public ebda_post
187public seg_40_value
188public hard_drive_post
189public int13_legacy
190public int70_handler
191public int75_handler
192public int15_handler32
193public int15_handler_mouse
194public iret_modify_cf
195public init_pic
196public floppy_post
197public int13_out
198public int13_disk
199public int13_notfloppy
200public int13_legacy
201public int13_noeltorito
202public int1c_handler
203public int10_handler
204public int74_handler
205public int76_handler
206public detect_parport
207public detect_serial
208public font8x8
209
210endif
211
212;; NOTE: The last 8K of the ROM BIOS are peppered with fixed locations which
213;; must be retained for compatibility. As a consequence, some of the space is
214;; going to be wasted, but the gaps should be filled with miscellaneous code
215;; and data when possible.
216
217SET_DEFAULT_CPU_286
218
219BIOSSEG segment 'CODE'
220 assume cs:BIOSSEG
221
222;;
223;; Start of fixed code - eoi_jmp_post is kept near here to allow short jumps.
224;;
225 BIOSORG 0E030h, 0E02Eh
226eoi_both_pics:
227 mov al, PIC_CMD_EOI
228 out PIC_SLAVE, al
229eoi_master_pic:
230 mov al, PIC_CMD_EOI
231 out PIC_MASTER, al
232 ret
233
234 ;; routine to write the pointer in DX:AX to memory starting
235 ;; at DS:BX (repeat CX times)
236 ;; - modifies BX, CX
237set_int_vects proc near
238
239 mov [bx], ax
240 mov [bx+2], dx
241 add bx, 4
242 loop set_int_vects
243 ret
244
245set_int_vects endp
246
247eoi_jmp_post:
248 call eoi_both_pics
249no_eoi_jmp_post:
250 xor ax, ax
251 mov ds, ax
252 jmp dword ptr ds:[0467h]
253
254seg_40_value: dw 40h ;; Replaces a push 40; pop ds.
255
256;; --------------------------------------------------------
257;; POST entry point
258;; --------------------------------------------------------
259 BIOSORG 0E05Bh, 0E059h
260post:
261 cli
262
263if VBOX_BIOS_CPU ge 80286
264 ;; Check if in protected (V86) mode. If so, the CPU needs
265 ;; to be reset.
266 .286p
267 smsw ax
268 test ax, 1
269 jz in_real_mode
270 SET_DEFAULT_CPU_286
271else
272 jmp in_real_mode
273endif
274
275 ;; Reset processor to get out of protected mode. Use system
276 ;; port instead of KBC.
277reset_sys:
278 mov al, 1
279 out 92h, al
280 jmp $ ; not strictly necessary in a VM
281
282
283in_real_mode:
284 ;; read the CMOS shutdown status
285 mov al, 0Fh
286 out CMOS_ADDR, al
287 in al, CMOS_DATA
288
289 ;; save status
290 xchg ah, al
291
292 ;; Check KBC self-test/shutdown flag. If it is set, we need
293 ;; to check for a reboot attempt.
294 in al, 64h
295 test al, 4 ; clear flag indicates cold boot
296 jz cont_post
297
298 ;; Warm boot, check the shutdown byte.
299 mov al, ah
300 or al, al
301 jnz cont_post
302
303 ;; Warm boot but shutdown byte is zero. This is either a warm
304 ;; boot request or an attempt to reset the system via triple
305 ;; faulting the CPU or similar. Check reboot flag.
306 ;; NB: At this point, registers need not be preserved.
307 mov ds, cs:[seg_40_value]
308 cmp word ptr ds:[72h], 1234h
309 jnz reset_sys ; trigger system reset
310
311cont_post:
312 ;; reset the shutdown status in CMOS
313 mov al, 0Fh
314 out CMOS_ADDR, al
315 mov al, 0
316 out CMOS_DATA, al
317
318 ;; pre-check the shutdown status - shutdown codes 9/A leave
319 ;; the hardware alone
320 mov al, ah
321 cmp al, 09h
322 jz check_shutdown
323 cmp al, 0Ah
324 jz check_shutdown
325
326 xor al, al
327
328 ;; reset the DMA controllers
329 out 00Dh, al
330 out 0DAh, al
331
332 ;; then initialize the DMA controllers
333 mov al, 0C0h
334 out 0D6h, al ; enable channel 4 cascade
335 mov al, 0
336 out 0D4h, al ; unmask channel 4
337
338check_shutdown:
339 ;; examine the shutdown status code
340 mov al, ah
341 cmp al, 0
342 jz normal_post
343
344 cmp al, 0Dh
345 jae normal_post
346 cmp al, 9
347 jne check_next_std
348 jmp return_blkmove
349check_next_std:
350
351 ;; 05h = EOI + jump through 40:67
352 cmp al, 5
353 je eoi_jmp_post
354 ;; 0ah = jump through 40:67 (no EOI) ;ba x 1 %fe05b ; ba x 1 %18b81
355 cmp al, 0ah
356 je no_eoi_jmp_post
357
358 ;; any other shutdown status values are ignored
359 ;; OpenSolaris sets the status to 0Ah in some cases?
360 jmp normal_post
361
362normal_post:
363 ;; shutdown code 0: normal startup
364
365 ;; Set up the stack top at 0:7800h. The stack should not be
366 ;; located above 0:7C00h; that conflicts with PXE, which
367 ;; considers anything above that address to be fair game.
368 ;; The traditional locations are 30:100 (PC) or 0:400 (PC/AT).
369 mov ax, 7800h
370 mov sp, ax
371 xor ax, ax
372 mov ds, ax
373 mov ss, ax
374
375 ;; clear the bottom of memory except for the word at 40:72
376 ;; TODO: Why not clear all of it? What's the point?
377 mov es, ax
378 xor di, di
379 cld
380 mov cx, 0472h / 2
381 rep stosw
382 inc di
383 inc di
384 mov cx, (1000h - 0472h - 2) / 2
385 rep stosw
386
387 ;; clear the remaining base memory except for the top
388 ;; of the EBDA (the MP table is planted there)
389 xor bx, bx
390memory_zero_loop:
391 add bx, 1000h
392 cmp bx, 9000h
393 jae memory_cleared
394 mov es, bx
395 xor di, di
396 mov cx, 8000h ; 32K words
397 rep stosw
398 jmp memory_zero_loop
399memory_cleared:
400 mov es, bx
401 xor di, di
402 mov cx, 7FF8h ; all but the last 16 bytes
403 rep stosw
404 xor bx, bx
405
406
407 C_SETUP
408 call _log_bios_start
409
410if VBOX_BIOS_CPU ge 80386
411 call pmode_setup
412endif
413
414 ;; set all interrupts in 00h-5Fh range to default handler
415 xor bx, bx
416 mov ds, bx
417 mov cx, 60h ; leave the rest as zeros
418 mov ax, dummy_iret
419 mov dx, BIOSSEG
420 call set_int_vects
421
422 ;; also set 68h-77h to default handler; note that the
423 ;; 60h-67h range must contain zeros for certain programs
424 ;; to function correctly
425 mov bx, 68h * 4
426 mov cx, 10h
427 call set_int_vects
428
429 ;; base memory in K to 40:13
430 mov ax, BASE_MEM_IN_K
431 mov ds:[413h], ax
432
433 ;; manufacturing test at 40:12
434 ;; zeroed out above
435
436 ;; set up various service vectors
437 ;; TODO: This should use the table at FEF3h instead
438 SET_INT_VECTOR 06h, BIOSSEG, int06_handler
439 SET_INT_VECTOR 11h, BIOSSEG, int11_handler
440 SET_INT_VECTOR 12h, BIOSSEG, int12_handler
441 SET_INT_VECTOR 15h, BIOSSEG, int15_handler
442 SET_INT_VECTOR 17h, BIOSSEG, int17_handler
443 SET_INT_VECTOR 18h, BIOSSEG, int18_handler
444 SET_INT_VECTOR 19h, BIOSSEG, int19_handler
445 SET_INT_VECTOR 1Ch, BIOSSEG, int1c_handler
446
447 call ebda_post
448
449 ;; Initialize PCI devices. This can and should be done early.
450if VBOX_BIOS_CPU ge 80386 ; (Impossible to do on 16-bit CPUs.)
451 call pcibios_init_iomem_bases
452 call pcibios_init_irqs
453endif
454 SET_INT_VECTOR 1Ah, BIOSSEG, int1a_handler
455
456 ;; PIT setup
457 SET_INT_VECTOR 08h, BIOSSEG, int08_handler
458 mov al, 34h ; timer 0, binary, 16-bit, mode 2
459 out 43h, al
460 mov al, 0 ; max count -> ~18.2 Hz
461 out 40h, al
462 out 40h, al
463
464 ;; video setup - must be done before POSTing VGA ROM
465 SET_INT_VECTOR 10h, BIOSSEG, int10_handler
466
467 ;; keyboard setup
468 SET_INT_VECTOR 09h, BIOSSEG, int09_handler
469 SET_INT_VECTOR 16h, BIOSSEG, int16_handler
470
471 xor ax, ax
472 mov ds, ax
473 ;; TODO: What's the point? The BDA is zeroed already?!
474 mov ds:[417h], al ; keyboard shift flags, set 1
475 mov ds:[418h], al ; keyboard shift flags, set 2
476 mov ds:[419h], al ; keyboard Alt-numpad work area
477 mov ds:[471h], al ; keyboard Ctrl-Break flag
478 mov ds:[497h], al ; keyboard status flags 4
479 mov al, 10h
480 mov ds:[496h], al ; keyboard status flags 3
481
482 mov bx, 1Eh
483 mov ds:[41Ah], bx ; keyboard buffer head
484 mov ds:[41Ch], bx ; keyboard buffer tail
485 mov ds:[480h], bx ; keyboard buffer start
486 mov bx, 3Eh
487 mov ds:[482h], bx ; keyboard buffer end
488
489 ;; store CMOS equipment byte in BDA
490 mov al, 14h
491 out CMOS_ADDR, al
492 in al, CMOS_DATA
493 mov ds:[410h], al
494
495 push ds
496 C_SETUP
497
498 ;; Scan for video ROMs in the C000-C800 range. This is done
499 ;; early so that errors are displayed on the screen.
500 mov ax, 0C000h
501 mov dx, 0C800h
502 call rom_scan_
503
504 ;; Initialize the keyboard
505 call _keyboard_init
506 pop ds
507
508 ;; parallel setup
509 SET_INT_VECTOR 0Fh, BIOSSEG, dummy_iret
510 xor ax, ax
511 mov ds, ax
512 xor bx, bx
513 mov cl, 14h ; timeout value
514 mov dx, 378h ; parallel port 1
515 call detect_parport
516 mov dx, 278h ; parallel port 2
517 call detect_parport
518 DO_shl bx, 0Eh
519 mov ax, ds:[410h] ; equipment word
520 and ax, 3FFFh
521 or ax, bx ; set number of parallel ports
522 mov ds:[410h], ax ; store in BDA
523
524 ;; Serial setup
525 SET_INT_VECTOR 0Bh, BIOSSEG, dummy_isr
526 SET_INT_VECTOR 0Ch, BIOSSEG, dummy_isr
527 SET_INT_VECTOR 14h, BIOSSEG, int14_handler
528 xor bx, bx
529 mov cl, 0Ah ; timeout value
530 mov dx, 3F8h ; first serial address
531 call detect_serial
532 mov dx, 2F8h ; second serial address
533 call detect_serial
534 mov dx, 3E8h ; third serial address
535 call detect_serial
536 mov dx, 2E8h ; fourth serial address
537 call detect_serial
538 DO_shl bx, 9
539 mov ax, ds:[410h] ; equipment word
540 and ax, 0F1FFh ; bits 9-11 determine serial ports
541 or ax, bx
542 mov ds:[410h], ax
543
544 ;; CMOS RTC
545 SET_INT_VECTOR 4Ah, BIOSSEG, dummy_iret ; TODO: redundant?
546 SET_INT_VECTOR 70h, BIOSSEG, int70_handler
547 ;; BIOS DATA AREA 4CEh ???
548 call rtc_post
549
550 jmp norm_post_cont
551
552
553;; --------------------------------------------------------
554;; NMI handler
555;; --------------------------------------------------------
556 BIOSORG 0E2C3h, 0E2C1h
557nmi:
558 C_SETUP
559 call _nmi_handler_msg
560 iret
561
562int75_handler:
563 out 0F0h, al ; clear IRQ13
564 call eoi_both_pics
565 int 2 ; emulate legacy NMI
566 iret
567
568
569hard_drive_post proc near
570
571 xor ax, ax
572 mov ds, ax
573 ;; TODO: Didn't we just clear the entire EBDA?
574 mov ds:[474h], al ; last HD operation status
575 mov ds:[477h], al ; HD port offset (XT only???)
576 mov ds:[48Ch], al ; HD status register
577 mov ds:[48Dh], al ; HD error register
578 mov ds:[48Eh], al ; HD task complete flag
579 mov al, 0C0h
580 mov ds:[476h], al ; HD control byte
581 ;; set up hard disk interrupt vectors
582 SET_INT_VECTOR 13h, BIOSSEG, int13_handler
583 SET_INT_VECTOR 76h, BIOSSEG, int76_handler
584 ;; INT 41h/46h: hard disk 0/1 dpt
585 ; TODO: This should be done from the code which
586 ; builds the DPTs?
587 SET_INT_VECTOR 41h, EBDA_SEG, 3Dh
588 SET_INT_VECTOR 46h, EBDA_SEG, 4Dh
589 ret
590
591hard_drive_post endp
592
593
594norm_post_cont:
595 ;; PS/2 mouse setup
596 SET_INT_VECTOR 74h, BIOSSEG, int74_handler
597
598 ;; IRQ 13h (FPU exception) setup
599 SET_INT_VECTOR 75h, BIOSSEG, int75_handler
600
601 call init_pic
602
603 C_SETUP
604
605if VBOX_BIOS_CPU ge 80386
606 ;; Set up local APIC
607 .386
608 pushad
609 call _apic_setup
610 popad
611 SET_DEFAULT_CPU_286
612endif
613
614 ;; ATA/ATAPI driver setup
615 call _ata_init
616 call _ata_detect
617
618ifdef VBOX_WITH_AHCI
619 ; AHCI driver setup
620 call _ahci_init
621endif
622
623ifdef VBOX_WITH_SCSI
624 ; SCSI driver setup
625 call _scsi_init
626endif
627
628 ;; floppy setup
629 call floppy_post
630
631 ;; hard drive setup
632 call hard_drive_post
633
634 C_SETUP ; in case assembly code changed things
635 ;; Scan for additional ROMs in the C800-EFFF range
636 mov ax, 0C800h
637 mov dx, 0F000h
638 call rom_scan_
639
640 call _print_bios_banner
641
642 ;; El Torito floppy/hard disk emulation
643 call _cdemu_init
644
645 ; TODO: what's the point of enabling interrupts here??
646 sti ; enable interrupts
647 int 19h
648 ;; does not return here
649 sti
650wait_forever:
651 hlt
652 jmp wait_forever
653 cli
654 hlt
655
656
657;;
658;; Return from block move (shutdown code 09h). Care must be taken to disturb
659;; register and memory state as little as possible.
660;;
661return_blkmove:
662 .286p
663 mov ax, 40h
664 mov ds, ax
665 ;; restore user stack
666 mov ss, ds:[69h]
667 mov sp, ds:[67h]
668 ;; reset A20 gate
669 in al, 92h
670 and al, 0FDh
671 out 92h, al
672 ;; ensure proper real mode IDT
673 lidt fword ptr cs:_rmode_IDT
674 ;; restore user segments
675 pop ds
676 pop es
677 ;; set up BP
678 mov bp, sp
679 ;; restore status code
680 in al, 80h
681 mov [bp+15], al
682 ;; set ZF/CF
683 cmp ah,al ; AH is zero here!
684 ;; restore registers and return
685 popa
686 sti
687 retf 2
688 SET_DEFAULT_CPU_286
689
690
691;; --------------------------------------------------------
692;; INT 13h handler - Disk services
693;; --------------------------------------------------------
694 BIOSORG 0E3FEh, 0E3FCh
695
696int13_handler:
697 jmp int13_relocated
698
699
700;; --------------------------------------------------------
701;; Fixed Disk Parameter Table
702;; --------------------------------------------------------
703 BIOSORG_CHECK 0E401h ; fixed wrt preceding
704
705rom_fdpt:
706
707;; --------------------------------------------------------
708;; INT 19h handler - Boot load service
709;; --------------------------------------------------------
710 BIOSORG 0E6F2h, 0E6F0h
711
712int19_handler:
713 jmp int19_relocated
714
715
716
717;; --------------------------------------------------------
718;; System BIOS Configuration Table
719;; --------------------------------------------------------
720 BIOSORG_CHECK 0E6F5h ; fixed wrt preceding
721; must match BIOS_CONFIG_TABLE
722bios_cfg_table:
723 dw 9 ; table size in bytes
724 db SYS_MODEL_ID
725 db SYS_SUBMODEL_ID
726 db BIOS_REVISION
727 ; Feature byte 1
728 ; b7: 1=DMA channel 3 used by hard disk
729 ; b6: 1=2 interrupt controllers present
730 ; b5: 1=RTC present
731 ; b4: 1=BIOS calls int 15h/4Fh for every key
732 ; b3: 1=wait for extern event supported (Int 15h/41h)
733 ; b2: 1=extended BIOS data area used
734 ; b1: 0=AT or ESDI bus, 1=MicroChannel
735 ; b0: 1=Dual bus (MicroChannel + ISA)
736ifdef BX_CALL_INT15_4F
737 db 74h; or USE_EBDA
738else
739 db 64h; or USE_EBDA
740endif
741 ; Feature byte 2
742 ; b7: 1=32-bit DMA supported
743 ; b6: 1=int16h, function 9 supported
744 ; b5: 1=int15h/C6h (get POS data) supported
745 ; b4: 1=int15h/C7h (get mem map info) supported
746 ; b3: 1=int15h/C8h (en/dis CPU) supported
747 ; b2: 1=non-8042 kb controller
748 ; b1: 1=data streaming supported
749 ; b0: reserved
750 db 40h
751 ; Feature byte 3
752 ; b7: not used
753 ; b6: reserved
754 ; b5: reserved
755 ; b4: POST supports ROM-to-RAM enable/disable
756 ; b3: SCSI on system board
757 ; b2: info panel installed
758 ; b1: Initial Machine Load (IML) system - BIOS on disk
759 ; b0: SCSI supported in IML
760 db 0
761 ; Feature byte 4
762 ; b7: IBM private
763 ; b6: EEPROM present
764 ; b5-3: ABIOS presence (011 = not supported)
765 ; b2: private
766 ; b1: memory split above 16Mb supported
767 ; b0: POSTEXT directly supported by POST
768 db 0
769 ; Feature byte 5 (IBM)
770 ; b1: enhanced mouse
771 ; b0: flash EPROM
772 db 0
773
774
775;; --------------------------------------------------------
776;; Baud Rate Generator Table
777;; --------------------------------------------------------
778 BIOSORG 0E729h, 0E727h
779
780
781;; --------------------------------------------------------
782;; INT 14h handler - Serial Communication Service
783;; --------------------------------------------------------
784 BIOSORG 0E739h, 0E737h
785int14_handler:
786 push ds
787 push es
788 DO_pusha
789 C_SETUP
790 call _int14_function
791 DO_popa
792 pop es
793 pop ds
794 iret
795
796
797
798;;
799;; Handler for unexpected hardware interrupts
800;;
801dummy_isr:
802 push ds
803 push es
804 DO_pusha
805 C_SETUP
806 call _dummy_isr_function
807 DO_popa
808 pop es
809 pop ds
810 iret
811
812
813init_pic proc near
814
815 mov al, 11h ; send init commands
816 out PIC_MASTER, al
817 out PIC_SLAVE, al
818 mov al, 08h ; base 08h
819 out PIC_MASTER+1, al
820 mov al, 70h ; base 70h
821 out PIC_SLAVE+1, al
822 mov al, 04h ; master PIC
823 out PIC_MASTER+1, al
824 mov al, 02h ; slave PIC
825 out PIC_SLAVE+1, al
826 mov al, 01h
827 out PIC_MASTER+1, al
828 out PIC_SLAVE+1, al
829 mov al, 0B8h ; unmask IRQs 0/1/2/6
830 out PIC_MASTER+1, al
831 mov al, 08Fh
832 out PIC_SLAVE+1, al ; unmask IRQs 12/13/14
833 ret
834
835init_pic endp
836
837ebda_post proc near
838
839 SET_INT_VECTOR 0Dh, BIOSSEG, dummy_isr ; IRQ 5
840 SET_INT_VECTOR 0Fh, BIOSSEG, dummy_isr ; IRQ 7
841 SET_INT_VECTOR 72h, BIOSSEG, dummy_isr ; IRQ 11
842 SET_INT_VECTOR 77h, BIOSSEG, dummy_isr ; IRQ 15
843
844 mov ax, EBDA_SEG
845 mov ds, ax
846 mov byte ptr ds:[0], EBDA_SIZE
847 ;; store EBDA seg in 40:0E
848 xor ax, ax
849 mov ds, ax
850 mov word ptr ds:[40Eh], EBDA_SEG
851 ret
852
853ebda_post endp
854
855
856
857;; --------------------------------------------------------
858;; INT 16h handler - Keyboard service
859;; --------------------------------------------------------
860 BIOSORG 0E82Eh, 0E82Ch
861int16_handler:
862 sti
863 push es
864 push ds
865 DO_pusha
866
867 cmp ah, 0
868 je int16_F00
869
870 cmp ah, 10h
871 je int16_F00
872
873 C_SETUP
874 call _int16_function
875 DO_popa
876 pop ds
877 pop es
878 iret
879
880int16_F00:
881 mov bx, 40h ; TODO: why 40h here and 0 elsewhere?
882 mov ds, bx
883int16_wait_for_key:
884 cli
885 mov bx, ds:[1Ah]
886 cmp bx, ds:[1Ch]
887 jne int16_key_found
888 sti
889 nop
890; TODO: review/enable?
891if 0
892 push ax
893 mov ax, 9002h
894 int 15h
895 pop ax
896endif
897 jmp int16_wait_for_key
898
899int16_key_found:
900 C_SETUP
901 call _int16_function
902 DO_popa
903 pop ds
904 pop es
905; TODO: review/enable? If so, flags should be restored here?
906if 0
907 push ax
908 mov ax, 9202h
909 int 15h
910 pop ax
911endif
912 iret
913
914
915if VBOX_BIOS_CPU ge 80386
916;; Quick and dirty protected mode entry/exit routines
917include pmode.inc
918
919;; Initialization code which needs to run in protected mode (LAPIC etc.)
920include pmsetup.inc
921endif
922
923
924KBDC_DISABLE EQU 0ADh
925KBDC_ENABLE EQU 0AEh
926KBC_CMD EQU 64h
927KBC_DATA EQU 60h
928
929;; --------------------------------------------------------
930;; INT 09h handler - Keyboard ISR (IRQ 1)
931;; --------------------------------------------------------
932 BIOSORG 0E987h, 0E985h
933int09_handler:
934 cli ; TODO: why? they're off already!
935 push ax
936 mov al, KBDC_DISABLE
937 out KBC_CMD, al
938
939 mov al, 0Bh
940 out PIC_MASTER, al
941 in al, PIC_MASTER
942 and al, 2
943 jz int09_finish
944
945 in al, KBC_DATA
946 push ds
947 DO_pusha
948 cld ; Before INT 15h (and any C code)
949ifdef BX_CALL_INT15_4F
950 mov ah, 4Fh
951 stc
952 int 15h ; keyboard intercept
953 jnc int09_done
954endif
955 sti ; Only after calling INT 15h
956
957 ;; check for extended key
958 cmp al, 0E0h
959 jne int09_check_pause
960 xor ax, ax
961 mov ds, ax
962 or byte ptr ds:[496h], 2 ; mf2_state |= 0x02
963 jmp int09_done
964
965int09_check_pause:
966 cmp al, 0E1h ; pause key?
967 jne int09_process_key
968 xor ax, ax
969 mov ds, ax
970 or byte ptr ds:[496h], 1 ; mf2_state | 0x01
971 jmp int09_done
972
973int09_process_key:
974 push es
975 C_SETUP
976 call _int09_function
977 pop es
978
979int09_done:
980 DO_popa
981 pop ds
982 cli
983 call eoi_master_pic
984
985int09_finish:
986 mov al, KBDC_ENABLE
987 out KBC_CMD, al
988 pop ax
989 iret
990
991
992;; --------------------------------------------------------
993;; INT 06h handler - Invalid Opcode Exception
994;; --------------------------------------------------------
995
996int06_handler:
997 DO_pusha
998 push es
999 push ds
1000 C_SETUP
1001 call _inv_op_handler
1002 pop ds
1003 pop es
1004 DO_popa
1005 iret
1006
1007;; --------------------------------------------------------
1008;; INT 13h handler - Diskette service
1009;; --------------------------------------------------------
1010 BIOSORG 0EC59h, 0EC57h
1011int13_diskette:
1012 jmp int13_noeltorito
1013
1014
1015
1016;; --------------------------------------------------------
1017;; INT 13h handler - Disk service
1018;; --------------------------------------------------------
1019int13_relocated:
1020 ;; check for an El-Torito function
1021 cmp ah, 4Ah
1022 jb int13_not_eltorito
1023
1024 cmp ah, 4Dh
1025 ja int13_not_eltorito
1026
1027 DO_pusha
1028 push es
1029 push ds
1030 C_SETUP ; TODO: setup C envrionment only once?
1031 DO_JMP_CALL_EX _int13_eltorito, int13_out, jmp_call_ret_int13_out ; ELDX not used
1032if VBOX_BIOS_CPU eq 8086
1033jmp_call_ret_int13_out: dw offset int13_out
1034endif
1035
1036int13_not_eltorito:
1037 push es
1038 push ax ; TODO: better register save/restore
1039 push bx
1040 push cx
1041 push dx
1042
1043 ;; check if emulation is active
1044 call _cdemu_isactive
1045 cmp al, 0
1046 je int13_cdemu_inactive
1047
1048 ;; check if access to the emulated drive
1049 call _cdemu_emulated_drive
1050 pop dx ; recover dx (destroyed by C code)
1051 push dx
1052 cmp al, dl ; INT 13h on emulated drive
1053 jne int13_nocdemu
1054
1055 pop dx
1056 pop cx
1057 pop bx
1058 pop ax
1059 pop es
1060
1061 DO_pusha
1062 push es
1063 push ds
1064 C_SETUP ; TODO: setup environment only once?
1065
1066 DO_JMP_CALL_EX _int13_cdemu, int13_out, jmp_call_ret_int13_out ; ELDX not used
1067
1068int13_nocdemu:
1069 and dl, 0E0h ; mask to get device class
1070 cmp al, dl
1071 jne int13_cdemu_inactive
1072
1073 pop dx
1074 pop cx
1075 pop bx
1076 pop ax
1077 pop es
1078
1079 push ax
1080 push cx
1081 push dx
1082 push bx
1083
1084 dec dl ; real drive is dl - 1
1085 jmp int13_legacy
1086
1087int13_cdemu_inactive:
1088 pop dx
1089 pop cx
1090 pop bx
1091 pop ax
1092 pop es
1093
1094int13_noeltorito:
1095 push ax
1096 push cx
1097 push dx
1098 push bx
1099int13_legacy:
1100 push dx ; push eltorito dx in place of sp
1101 push bp
1102 push si
1103 push di
1104 push es
1105 push ds
1106 C_SETUP ; TODO: setup environment only once?
1107
1108 ;; now the registers can be restored with
1109 ;; pop ds; pop es; DO_popa; iret
1110 test dl, 80h ; non-removable?
1111 jnz int13_notfloppy
1112
1113 DO_JMP_CALL_EX _int13_diskette_function, int13_out, jmp_call_ret_int13_out
1114
1115int13_notfloppy:
1116 cmp dl, 0E0h
1117 jb int13_notcdrom
1118
1119 ;; ebx may be modified, save here
1120 ;; TODO: check/review 32-bit register use
1121 ;; @todo figure if 80286/8086 variant is applicable.
1122 .386
1123 shr ebx, 16
1124 push bx
1125 call _int13_cdrom
1126 pop bx
1127 shl ebx, 16
1128 SET_DEFAULT_CPU_286
1129 jmp int13_out
1130
1131int13_notcdrom:
1132int13_disk:
1133 cmp ah,40h
1134 ja int13x
1135 call _int13_harddisk
1136 jmp int13_out
1137
1138int13x:
1139 call _int13_harddisk_ext
1140
1141int13_out:
1142 pop ds
1143 pop es
1144 DO_popa
1145 iret
1146
1147
1148
1149; parallel port detection: port in dx, index in bx, timeout in cl
1150detect_parport proc near
1151
1152 push dx
1153 inc dx
1154 inc dx
1155 in al, dx
1156 and al, 0DFh ; clear input mode
1157 out dx, al
1158 pop dx
1159 mov al, 0AAh
1160 out dx, al
1161 in al, dx
1162 cmp al, 0AAh
1163 jne no_parport
1164
1165 push bx
1166 shl bx, 1
1167 mov [bx+408h], dx ; parallel I/O address
1168 pop bx
1169 mov [bx+478h], cl ; parallel printer timeout
1170 inc bx
1171no_parport:
1172 ret
1173
1174detect_parport endp
1175
1176; setial port detection: port in dx, index in bx, timeout in cl
1177detect_serial proc near
1178
1179 push dx
1180 inc dx
1181 mov al, 2
1182 out dx, al
1183 in al, dx
1184 cmp al, 2
1185 jne no_serial
1186
1187 inc dx
1188 in al, dx
1189 cmp al, 2
1190 jne no_serial
1191
1192 dec dx
1193 xor al, al
1194 pop dx
1195 push bx
1196 shl bx, 1
1197 mov [bx+400h], dx ; serial I/O address
1198 pop bx
1199 mov [bx+47Ch], cl ; serial timeout
1200 inc bx
1201 ret
1202
1203no_serial:
1204 pop dx
1205 ret
1206
1207detect_serial endp
1208
1209
1210;;
1211;; POST: Floppy drive
1212;;
1213floppy_post proc near
1214
1215 xor ax, ax
1216 mov ds, ax
1217
1218 ;; TODO: This code is really stupid. Zeroing the BDA byte
1219 ;; by byte is dumb, and it's been already zeroed elsewhere!
1220 mov al, 0
1221 mov ds:[43Eh], al ; drive 0/1 uncalibrated, no IRQ
1222 mov ds:[43Fh], al ; motor status
1223 mov ds:[440h], al ; motor timeout counter
1224 mov ds:[441h], al ; controller status return code
1225 mov ds:[442h], al ; hd/floppy ctlr status register
1226 mov ds:[443h], al ; controller status register 1
1227 mov ds:[444h], al ; controller status register 2
1228 mov ds:[445h], al ; cylinder number
1229 mov ds:[446h], al ; head number
1230 mov ds:[447h], al ; sector number
1231 mov ds:[448h], al ; bytes written
1232
1233 mov ds:[48Bh], al ; configuration data
1234
1235 mov al, 10h ; floppy drive type
1236 out CMOS_ADDR, al
1237 in al, CMOS_DATA
1238 mov ah, al ; save drive type byte
1239
1240look_drive0:
1241 ; TODO: pre-init bl to reduce jumps
1242 DO_shr al, 4 ; drive 0 in high nibble
1243 jz f0_missing ; jump if no drive
1244 mov bl, 7 ; drv0 determined, multi-rate, chgline
1245 jmp look_drive1
1246
1247f0_missing:
1248 mov bl, 0 ; no drive 0
1249
1250look_drive1:
1251 mov al, ah ; restore CMOS data
1252 and al, 0Fh ; drive 1 in low nibble
1253 jz f1_missing
1254 or bl, 70h ; drv1 determined, multi-rate, chgline
1255f1_missing:
1256 mov ds:[48Fh], bl ; store in BDA
1257
1258 ;; TODO: See above. Dumb *and* redundant!
1259 mov al, 0
1260 mov ds:[490h], al ; drv0 media state
1261 mov ds:[491h], al ; drv1 media state
1262 mov ds:[492h], al ; drv0 operational state
1263 mov ds:[493h], al ; drv1 operational state
1264 mov ds:[494h], al ; drv0 current cylinder
1265 mov ds:[495h], al ; drv1 current cylinder
1266
1267 mov al, 2
1268 out 0Ah, al ; unmask DMA channel 2
1269
1270 SET_INT_VECTOR 1Eh, BIOSSEG, _diskette_param_table
1271 SET_INT_VECTOR 40h, BIOSSEG, int13_diskette
1272 SET_INT_VECTOR 0Eh, BIOSSEG, int0e_handler ; IRQ 6
1273
1274 ret
1275
1276floppy_post endp
1277
1278
1279bcd_to_bin proc near
1280
1281 ;; in : AL in packed BCD format
1282 ;; out: AL in binary, AH always 0
1283if VBOX_BIOS_CPU ge 80186
1284 shl ax, 4
1285 shr al, 4
1286else
1287 push cx
1288 mov cl, 4
1289 shl ax, cl
1290 shr al, cl
1291 pop cx
1292endif
1293 aad
1294 ret
1295
1296bcd_to_bin endp
1297
1298rtc_post proc near
1299
1300if VBOX_BIOS_CPU lt 80386 ;; @todo fix loopy code below
1301 ;; get RTC seconds
1302 mov al, 0
1303 out CMOS_ADDR, al
1304 in al, CMOS_DATA ; RTC seconds, in BCD
1305 call bcd_to_bin ; ax now has seconds in binary
1306 test al, al
1307 xor ah, ah
1308 mov dx, 0x1234 ; 18206507*0x100/1000000 = 0x1234 (4660.865792)
1309 mul dx
1310 mov cx, ax ; tick count in dx:cx
1311
1312 ;; get RTC minutes
1313 mov al, 2
1314 out CMOS_ADDR, al
1315 in al, CMOS_DATA ; RTC minutes, in BCD
1316 call bcd_to_bin ; eax now has minutes in binary
1317 test al, al
1318 jz rtc_post_hours
1319rtc_pos_min_loop: ; 18206507*60*0x100/1000000 = 0x44463 (279651.94752)
1320 add cx, 0x4463
1321 adc dx, 0x0004
1322 dec al
1323 jnz rtc_pos_min_loop
1324
1325 ;; get RTC hours
1326rtc_post_hours:
1327 mov al, 4
1328 out CMOS_ADDR, al
1329 in al, CMOS_DATA ; RTC hours, in BCD
1330 call bcd_to_bin ; eax now has hours in binary
1331 test al, al
1332 jz rtc_pos_shift
1333rtc_pos_hour_loop: ; 18206507*3600*0x100/1000000 = 0x100076C (16779116.8512)
1334 add cx, 0x076C
1335 adc dx, 0x0100
1336 dec al
1337 jnz rtc_pos_hour_loop
1338
1339rtc_pos_shift:
1340 mov cl, ch
1341 mov ch, dl
1342 mov dl, dh
1343 xor dh, dh
1344 mov ds:[46Ch], cx ; timer tick count
1345 mov ds:[46Ch+2], dx ; timer tick count
1346 mov ds:[470h], dh ; rollover flag
1347
1348else
1349 .386
1350 ;; get RTC seconds
1351 xor eax, eax
1352 mov al, 0
1353 out CMOS_ADDR, al
1354 in al, CMOS_DATA ; RTC seconds, in BCD
1355 call bcd_to_bin ; eax now has seconds in binary
1356 mov edx, 18206507
1357 mul edx
1358 mov ebx, 1000000
1359 xor edx, edx
1360 div ebx
1361 mov ecx, eax ; total ticks in ecx
1362
1363 ;; get RTC minutes
1364 xor eax, eax
1365 mov al, 2
1366 out CMOS_ADDR, al
1367 in al, CMOS_DATA ; RTC minutes, in BCD
1368 call bcd_to_bin ; eax now has minutes in binary
1369 mov edx, 10923904
1370 mul edx
1371 mov ebx, 10000
1372 xor edx, edx
1373 div ebx
1374 add ecx, eax ; add to total ticks
1375
1376 ;; get RTC hours
1377 xor eax, eax
1378 mov al, 4
1379 out CMOS_ADDR, al
1380 in al, CMOS_DATA ; RTC hours, in BCD
1381 call bcd_to_bin ; eax now has hours in binary
1382 mov edx, 65543427
1383 mul edx
1384 mov ebx, 1000
1385 xor edx, edx
1386 div ebx
1387 add ecx, eax ; add to total ticks
1388
1389 mov ds:[46Ch], ecx ; timer tick count
1390 xor al, al ; TODO: redundant?
1391 mov ds:[470h], al ; rollover flag
1392 .286
1393endif
1394 ret
1395
1396rtc_post endp
1397
1398
1399
1400;; --------------------------------------------------------
1401;; INT 0Eh handler - Diskette IRQ 6 ISR
1402;; --------------------------------------------------------
1403 BIOSORG 0EF57h, 0EF55h
1404int0e_handler:
1405 push ax
1406 push dx
1407 mov dx, 3F4h
1408 in al, dx
1409 and al, 0C0h
1410 cmp al, 0C0h
1411 je int0e_normal
1412 mov dx, 3F5h
1413 mov al, 08h ; sense interrupt
1414 out dx, al
1415int0e_loop1:
1416 mov dx, 3F4h ; TODO: move out of the loop?
1417 in al, dx
1418 and al, 0C0h
1419 cmp al, 0C0h
1420 jne int0e_loop1
1421
1422int0e_loop2:
1423 mov dx, 3F5h ; TODO: inc/dec dx instead
1424 in al, dx
1425 mov dx, 3F4h
1426 in al, dx
1427 and al, 0C0h
1428 cmp al, 0C0h
1429 je int0e_loop2
1430
1431int0e_normal:
1432 push ds
1433 xor ax, ax
1434 mov ds, ax
1435 call eoi_master_pic
1436 ; indicate that an interrupt occurred
1437 or byte ptr ds:[43Eh], 80h
1438 pop ds
1439 pop dx
1440 pop ax
1441 iret
1442
1443
1444;; --------------------------------------------------------
1445;; Diskette Parameter Table
1446;; --------------------------------------------------------
1447 BIOSORG 0EFC7h, 0EFC5h
1448_diskette_param_table:
1449 db 0AFh
1450 db 2 ; HLT=1, DMA mode
1451 db 025h
1452 db 2
1453 db 18 ; SPT (good for 1.44MB media)
1454 db 01Bh
1455 db 0FFh
1456 db 06Ch
1457 db 0F6h ; format filler
1458 db 15
1459 db 8
1460
1461
1462
1463;; --------------------------------------------------------
1464;; INT 17h handler - Printer service
1465;; --------------------------------------------------------
1466 BIOSORG_CHECK 0EFD2h ; fixed WRT preceding code
1467
1468 jmp int17_handler ; NT floppy boot workaround
1469 ; see @bugref{6481}
1470int17_handler:
1471 push ds
1472 push es
1473 DO_pusha
1474 C_SETUP
1475 call _int17_function
1476 DO_popa
1477 pop es
1478 pop ds
1479 iret
1480
1481
1482
1483;; Protected mode IDT descriptor
1484;;
1485;; The limit is 0 to cause a shutdown if an exception occurs
1486;; in protected mode. TODO: Is that what we really want?
1487;;
1488;; Set base to F0000 to correspond to beginning of BIOS,
1489;; in case an IDT is defined later.
1490
1491_pmode_IDT:
1492 dw 0 ; limit 15:0
1493 dw 0 ; base 15:0
1494 dw 0Fh ; base 23:16
1495
1496
1497;; Real mode IDT descriptor
1498;;
1499;; Set to typical real-mode values.
1500;; base = 000000
1501;; limit = 03ff
1502
1503_rmode_IDT:
1504 dw 3FFh ; limit 15:00
1505 dw 0 ; base 15:00
1506 dw 0 ; base 23:16
1507
1508
1509;;
1510;; INT 1Ch
1511;;
1512;; TODO: Why does this need a special handler?
1513int1c_handler: ;; user timer tick
1514 iret
1515
1516
1517
1518;; --------------------------------------------------------
1519;; INT 10h functions 0-Fh entry point
1520;; --------------------------------------------------------
1521 BIOSORG 0F045h, 0F043h
1522i10f0f_entry:
1523 iret
1524
1525
1526;; --------------------------------------------------------
1527;; INT 10h handler - MDA/CGA video
1528;; --------------------------------------------------------
1529 BIOSORG 0F065h, 0F063h
1530int10_handler:
1531 ;; do nothing - assumes VGA
1532 iret
1533
1534
1535;; --------------------------------------------------------
1536;; MDA/CGA Video Parameter Table (INT 1Dh)
1537;; --------------------------------------------------------
1538 BIOSORG 0F0A4h, 0F0A2h
1539mdacga_vpt:
1540
1541
1542;;
1543;; INT 18h - boot failure
1544;;
1545int18_handler:
1546 C_SETUP
1547 call _int18_panic_msg
1548 ;; TODO: handle failure better?
1549 hlt
1550 iret
1551
1552;;
1553;; INT 19h - boot service - relocated
1554;;
1555int19_relocated:
1556; If an already booted OS calls int 0x19 to reboot, it is not sufficient
1557; just to try booting from the configured drives. All BIOS variables and
1558; interrupt vectors need to be reset, otherwise strange things may happen.
1559; The approach used is faking a warm reboot (which just skips showing the
1560; logo), which is a bit more than what we need, but hey, it's fast.
1561 mov bp, sp
1562 mov ax, [bp+2] ; TODO: redundant? address via sp?
1563 cmp ax, BIOSSEG ; check caller's segment
1564 jz bios_initiated_boot
1565
1566 xor ax, ax
1567 mov ds, ax
1568 mov ax, 1234h
1569 mov ds:[472], ax
1570 jmp post
1571
1572bios_initiated_boot:
1573 ;; The C worker function returns the boot drive in bl and
1574 ;; the boot segment in ax. In case of failure, the boot
1575 ;; segment will be zero.
1576 C_SETUP ; TODO: Here? Now?
1577 push bp
1578 mov bp, sp
1579
1580 ;; 1st boot device
1581 mov ax, 1
1582 push ax
1583 call _int19_function
1584 inc sp
1585 inc sp
1586 test ax, ax ; if 0, try next device
1587 jnz boot_setup
1588
1589 ;; 2nd boot device
1590 mov ax, 2
1591 push ax
1592 call _int19_function
1593 inc sp
1594 inc sp
1595 test ax, ax ; if 0, try next device
1596 jnz boot_setup
1597
1598 ; 3rd boot device
1599 mov ax, 3
1600 push ax
1601 call _int19_function
1602 inc sp
1603 inc sp
1604 test ax, ax ; if 0, try next device
1605 jnz boot_setup
1606
1607 ; 4th boot device
1608 mov ax, 4
1609 push ax
1610 call _int19_function
1611 inc sp
1612 inc sp
1613 test ax, ax ; if 0, invoke INT 18h
1614 jz int18_handler
1615
1616boot_setup:
1617; TODO: the drive should be in dl already??
1618;; mov dl, bl ; tell guest OS what boot drive is
1619if VBOX_BIOS_CPU lt 80386
1620 mov [bp], ax
1621 DO_shl ax, 4
1622 mov [bp+2], ax ; set ip
1623 mov ax, [bp]
1624else
1625 .386 ; NB: We're getting garbage into high eax bits
1626 shl eax, 4 ; convert seg to ip
1627 mov [bp+2], ax ; set ip
1628
1629 shr eax, 4 ; get cs back
1630 .286
1631endif
1632 and ax, BIOSSEG ; remove what went in ip
1633 mov [bp+4], ax ; set cs
1634 xor ax, ax
1635 mov ds, ax
1636 mov es, ax
1637 mov [bp], ax ; TODO: what's this?!
1638 mov ax, 0AA55h ; set ok flag ; TODO: and this?
1639
1640 pop bp ; TODO: why'd we just zero it??
1641 iret ; beam me up scotty
1642
1643;; PCI BIOS
1644
1645include pcibios.inc
1646include pirq.inc
1647
1648
1649;; --------------------------------------------------------
1650;; INT 12h handler - Memory size
1651;; --------------------------------------------------------
1652 BIOSORG 0F841h, 0F83Fh
1653int12_handler:
1654 ;; Don't touch - fixed size!
1655 sti
1656 push ds
1657 mov ax, 40h
1658 mov ds, ax
1659 mov ax, ds:[13h]
1660 pop ds
1661 iret
1662
1663
1664;; --------------------------------------------------------
1665;; INT 11h handler - Equipment list service
1666;; --------------------------------------------------------
1667 BIOSORG_CHECK 0F84Dh ; fixed wrt preceding code
1668int11_handler:
1669 ;; Don't touch - fixed size!
1670 sti
1671 push ds
1672 mov ax, 40h
1673 mov ds, ax
1674 mov ax, ds:[10h]
1675 pop ds
1676 iret
1677
1678
1679;; --------------------------------------------------------
1680;; INT 15h handler - System services
1681;; --------------------------------------------------------
1682 BIOSORG_CHECK 0F859h ; fixed wrt preceding code
1683int15_handler:
1684
1685if VBOX_BIOS_CPU ge 80286
1686 cmp ah, 87h
1687 jne not_blkmove
1688
1689 ;; INT 15h/87h has semi-public interface because software
1690 ;; may use CMOS shutdown status code 9 for its own purposes.
1691 ;; The stack layout has to match.
1692 pusha
1693 push es
1694 push ds
1695 C_SETUP
1696 call _int15_blkmove
1697 pop ds
1698 pop es
1699 popa
1700 iret
1701not_blkmove:
1702
1703endif
1704
1705 pushf
1706 push ds
1707 push es
1708 C_SETUP
1709 cmp ah, 86h
1710 je int15_handler32
1711 cmp ah, 0E8h
1712 je int15_handler32
1713 cmp ah, 0d0h
1714 je int15_handler32
1715 DO_pusha
1716 cmp ah, 53h ; APM function?
1717 je apm_call
1718 cmp ah, 0C2h ; PS/2 mouse function?
1719 je int15_handler_mouse
1720
1721 call _int15_function
1722int15_handler_popa_ret:
1723 DO_popa
1724int15_handler32_ret:
1725 pop es
1726 pop ds
1727 popf
1728 jmp iret_modify_cf
1729
1730apm_call:
1731 call _apm_function
1732 jmp int15_handler_popa_ret
1733
1734int15_handler_mouse:
1735 call _int15_function_mouse
1736 jmp int15_handler_popa_ret
1737
1738int15_handler32:
1739if VBOX_BIOS_CPU ge 80386
1740 ;; need to save/restore 32-bit registers
1741 .386
1742 pushad
1743 call _int15_function32
1744 popad
1745 .286
1746else
1747 DO_pusha
1748 call _int15_function32
1749 DO_popa
1750endif
1751 jmp int15_handler32_ret
1752
1753;;
1754;; Perform an IRET but retain the current carry flag value
1755;;
1756iret_modify_cf:
1757 jc carry_set
1758 push bp
1759 mov bp, sp
1760 and byte ptr [bp + 6], 0FEh
1761 pop bp
1762 iret
1763carry_set:
1764 push bp
1765 mov bp, sp
1766 or byte ptr [bp + 6], 1
1767 pop bp
1768 iret
1769
1770;;
1771;; INT 74h handler - PS/2 mouse (IRQ 12)
1772;;
1773int74_handler proc
1774
1775 sti
1776 DO_pusha
1777 push es
1778 push ds
1779 xor ax, ax
1780 push ax ; placeholder for status
1781 push ax ; placeholder for X
1782 push ax ; placeholder for Y
1783 push ax ; placeholder for Z
1784 push ax ; placeholder for make_far_call bool
1785 C_SETUP
1786 call _int74_function
1787 pop cx ; pop make_far_call flag
1788 jcxz int74_done
1789
1790 ;; make far call to EBDA:0022
1791if VBOX_BIOS_CPU ge 80186
1792 push 0
1793else
1794 xor ax, ax
1795 push ax
1796endif
1797 pop ds
1798 push ds:[40Eh]
1799 pop ds
1800 call far ptr ds:[22h]
1801int74_done:
1802 cli
1803 call eoi_both_pics
1804 add sp, 8 ; remove status, X, Y, Z
1805 pop ds
1806 pop es
1807 DO_popa
1808 iret
1809
1810int74_handler endp
1811
1812int76_handler proc
1813
1814 ;; record completion in BIOS task complete flag
1815 push ax
1816 push ds
1817 mov ax, 40h
1818 mov ds, ax
1819 mov byte ptr ds:[8Eh], 0FFh
1820 call eoi_both_pics
1821 pop ds
1822 pop ax
1823 iret
1824
1825int76_handler endp
1826
1827
1828;;
1829;; IRQ 8 handler (RTC)
1830;;
1831int70_handler:
1832 push es
1833 push ds
1834 DO_pusha
1835 C_SETUP
1836 call _int70_function
1837 DO_popa
1838 pop ds
1839 pop es
1840 iret
1841
1842
1843
1844if VBOX_BIOS_CPU lt 80386
1845;
1846; We're tight on space down below in the int08_handler, so put
1847; the 16-bit rollover code here.
1848;
1849int08_maybe_rollover:
1850 ja int08_rollover
1851 cmp ax, 00B0h
1852 jb int08_rollover_store
1853 ;; there has been a midnight rollover
1854int08_rollover:
1855 xor dx, dx
1856 xor ax, ax
1857
1858 inc byte ptr ds:[70h] ; increment rollover flag
1859int08_rollover_store:
1860 jmp int08_store_ticks
1861endif
1862
1863
1864;; --------------------------------------------------------
1865;; 8x8 font (first 128 characters)
1866;; --------------------------------------------------------
1867 BIOSORG 0FA6Eh, 0FA6Ch
1868include font8x8.inc
1869
1870
1871;; --------------------------------------------------------
1872;; INT 1Ah handler - Time of the day + PCI BIOS
1873;; --------------------------------------------------------
1874 BIOSORG_CHECK 0FE6Eh ; fixed wrt preceding table
1875int1a_handler:
1876if VBOX_BIOS_CPU ge 80386
1877 cmp ah, 0B1h
1878 jne int1a_normal
1879
1880 push es
1881 push ds
1882 C_SETUP
1883 .386
1884 pushad
1885 call _pci16_function
1886 popad
1887 .286
1888 pop ds
1889 pop es
1890 iret
1891endif
1892
1893int1a_normal:
1894 push es
1895 push ds
1896 DO_pusha
1897 C_SETUP
1898int1a_callfunction:
1899 call _int1a_function
1900 DO_popa
1901 pop ds
1902 pop es
1903 iret
1904
1905
1906;; --------------------------------------------------------
1907;; Timer tick - IRQ 0 handler
1908;; --------------------------------------------------------
1909 BIOSORG 0FEA5h, 0FEA3h
1910int08_handler:
1911if VBOX_BIOS_CPU ge 80386
1912 .386
1913 sti
1914 push eax
1915else
1916 sti
1917 push ax
1918endif
1919 push ds
1920 push dx
1921 mov ax, 40h
1922 mov ds, ax
1923
1924if VBOX_BIOS_CPU ge 80386
1925 mov eax, ds:[6Ch] ; get ticks dword
1926 inc eax
1927else
1928 mov ax, ds:[6Ch] ; get ticks dword
1929 mov dx, ds:[6Ch+2]
1930 inc ax ; inc+jz+inc saves two bytes over add+adc.
1931 jnz int08_compare
1932 inc dx
1933int08_compare:
1934endif
1935
1936 ;; compare eax to one day's worth of ticks (at 18.2 Hz)
1937if VBOX_BIOS_CPU ge 80386
1938 cmp eax, 1800B0h
1939 jb int08_store_ticks
1940else
1941 cmp dx, 18h
1942 jb int08_store_ticks
1943 jmp int08_maybe_rollover
1944endif
1945
1946if VBOX_BIOS_CPU ge 80386
1947 ;; there has been a midnight rollover
1948 xor eax, eax
1949 inc byte ptr ds:[70h] ; increment rollover flag
1950
1951int08_store_ticks:
1952 mov ds:[6Ch], eax
1953else
1954int08_store_ticks:
1955 mov ds:[6Ch], ax
1956 mov ds:[6Ch+2], dx
1957endif
1958
1959 ;; time to turn off floppy drive motor(s)?
1960 mov al, ds:[40h]
1961 or al, al
1962 jz int08_floppy_off
1963 dec al
1964 mov ds:[40h], al
1965 jnz int08_floppy_off
1966 ;; turn motor(s) off
1967 mov dx, 03F2h
1968 in al, dx
1969 and al, 0CFh
1970 out dx, al
1971int08_floppy_off:
1972
1973 int 1Ch ; call the user timer handler
1974
1975 cli
1976 call eoi_master_pic
1977 pop dx
1978 pop ds
1979if VBOX_BIOS_CPU ge 80386
1980 pop eax
1981 .286
1982else
1983 pop ax
1984endif
1985 iret
1986
1987
1988;; --------------------------------------------------------
1989;; Initial interrupt vector offsets for POST
1990;; --------------------------------------------------------
1991 BIOSORG 0FEF3h, 0FEF1h
1992vector_table:
1993
1994
1995
1996;; --------------------------------------------------------
1997;; BIOS copyright string
1998;; --------------------------------------------------------
1999 BIOSORG 0FF00h, 0FEFEh
2000bios_string:
2001 db BIOS_COPYRIGHT
2002
2003
2004;; --------------------------------------------------------
2005;; IRET - default interrupt handler
2006;; --------------------------------------------------------
2007 BIOSORG 0FF53h, 0FF51h
2008
2009dummy_iret:
2010 iret
2011
2012
2013;; --------------------------------------------------------
2014;; INT 05h - Print Screen service
2015;; --------------------------------------------------------
2016 BIOSORG_CHECK 0FF54h ; fixed wrt preceding
2017int05_handler:
2018 ;; Not implemented
2019 iret
2020
2021include smidmi.inc
2022
2023;; --------------------------------------------------------
2024;; Processor reset entry point
2025;; --------------------------------------------------------
2026 BIOSORG 0FFF0h, 0FFEEh
2027cpu_reset:
2028 ;; This is where the CPU starts executing after a reset
2029 jmp far ptr post
2030
2031 ;; BIOS build date
2032 db BIOS_BUILD_DATE
2033 db 0 ; padding
2034 ;; System model ID
2035 db SYS_MODEL_ID
2036 ;; Checksum byte
2037 db 0FFh
2038
2039
2040BIOSSEG ends
2041
2042 end
2043
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