summaryrefslogtreecommitdiff
path: root/boot/boot.S
diff options
context:
space:
mode:
authorFreya Murphy <freya@freyacat.org>2025-03-25 17:36:52 -0400
committerFreya Murphy <freya@freyacat.org>2025-03-25 17:38:22 -0400
commit6af21e6a4f2251e71353562d5df7f376fdffc270 (patch)
treede20c7afc9878422c81e34f30c6b010075e9e69a /boot/boot.S
downloadcomus-6af21e6a4f2251e71353562d5df7f376fdffc270.tar.gz
comus-6af21e6a4f2251e71353562d5df7f376fdffc270.tar.bz2
comus-6af21e6a4f2251e71353562d5df7f376fdffc270.zip
initial checkout from wrc
Diffstat (limited to 'boot/boot.S')
-rw-r--r--boot/boot.S666
1 files changed, 666 insertions, 0 deletions
diff --git a/boot/boot.S b/boot/boot.S
new file mode 100644
index 0000000..50d6188
--- /dev/null
+++ b/boot/boot.S
@@ -0,0 +1,666 @@
+/**
+** SCCS ID: @(#)boot.S 2.4 1/22/25
+**
+** @file boot.S
+**
+** @author Jon Coles
+** copyleft 1999 Jon Coles
+**
+** @author Warren R. Carithers, K. Reek, Garrett C. Smith
+** @author Walter Litwinczyk, David C. Larsen, Sean T. Congden
+**
+** Bootstrap routine.
+**
+** This bootstrap program is loaded by the PC BIOS into memory at
+** location 0000:7C00. It must be exactly 512 bytes long, and must
+** end with the hex sequence AA55 at location 1FE.
+**
+** The bootstrap initially sets up a stack in low memory. Next, it
+** loads a second sector at 0000:7E00 (immediately following the
+** boot block). Then it loads the target program at TARGET_ADDR,
+** switches to protected mode, and branches to the target program.
+**
+** NOTE: To zero out the BSS segment, define CLEAR_BSS when this code
+** is assembled.
+**
+** Must assemble this as 16-bit code.
+*/
+ .code16
+
+#define ASM_SRC
+
+#include <bootstrap.h>
+#include <x86/bios.h>
+#include <x86/arch.h>
+
+/*
+** Symbol for locating the beginning of the code.
+*/
+ .globl bootentry
+
+ .text
+bootentry:
+
+/*
+** Entry point. Disable interrupts and set up a runtime stack.
+*/
+ cli
+
+ movw $BOOT_SEG, %ax /* data seg. base address */
+ movw %ax, %ds
+ movw %ax, %ss /* also stack seg. base */
+ movw $BOOT_SP_DISP, %ax
+ movw %ax, %sp
+
+/*
+** Next, verify that the disk is there and working.
+*/
+ movb $BD_CHECK, %ah /* test the disk status and make sure */
+ movb drive, %dl /* it's safe to proceed */
+ int $BIOS_DISK
+ jnc diskok
+
+ movw $err_diskstatus, %si /* Something went wrong; print a message */
+ call dispMsg /* and freeze. */
+ jmp .
+
+/*
+** The disk is there. Reset it, and retrieve the disk parameters.
+*/
+diskok:
+ movw $BD_RESET, %ax /* Reset the disk */
+ movb drive, %dl
+ int $BIOS_DISK
+
+ /* determine number of heads and sectors/track */
+ xorw %ax, %ax /* set ES:DI = 0000:0000 in case of BIOS bugs */
+ movw %ax, %es
+ movw %ax, %di
+ movb $BD_PARAMS, %ah /* get drive parameters */
+ movb drive, %dl /* hard disk or floppy */
+ int $BIOS_DISK
+
+ /* store (max + 1) - CL[5:0] = maximum head, DH = maximum head */
+ andb $0x3F, %cl
+ incb %cl
+ incb %dh
+
+ movb %cl, max_sec
+ movb %dh, max_head
+
+/*
+** The disk is OK, so we now need to load the second half of the bootstrap.
+** It must immediately follow the boot sector on the disk, and the target
+** program(s) must immediately follow that.
+*/
+ movw $msg_loading, %si /* Print the Loading message */
+ call dispMsg
+
+ movw $1, %ax /* sector count = 1 */
+ movw $BOOT_SEG, %bx /* read this into memory that */
+ movw %bx, %es /* immediately follows this code. */
+ movw $PART2_DISP, %bx
+ call readprog
+
+/*
+** We've got the second block of the bootstrap program in memory. Now
+** read all of the user's program blocks. Use %di to point to the
+** count field for the next block to load.
+*/
+ movw $k_sect, %di
+
+ pushw %ds
+ movw (%di), %bx
+ movw $MMAP_SEG, %ax
+ movw %ax, %ds
+ movw %bx, MMAP_SECTORS /* store kernel image size */
+ popw %ds
+
+/*
+** Each target program has three values in the array at the end of the
+** second half of the bootstrap: the offset and segment base address
+** where the program should go, and the sector count.
+*/
+nextblock:
+ movw (%di), %ax /* get the # of sectors */
+ testw %ax, %ax /* is it zero? */
+ jz done_loading /* yes, nothing more to load. */
+
+ subw $2, %di
+ movw (%di), %bx /* get the segment value */
+ movw %bx, %es /* and copy it to %es */
+ subw $2, %di
+ movw (%di), %bx /* get the address offset */
+ subw $2, %di
+ pushw %di /* save di */
+ call readprog /* read this program block, */
+ popw %di /* and restore di */
+ jmp nextblock /* then go back and read the next one. */
+
+/*
+** Read one complete program block into memory.
+**
+** ax: number of sectors to read
+** es:bx = starting address for the block
+*/
+readprog:
+ pushw %ax /* save sector count */
+
+ movw $3, %cx /* initial retry count is 3 */
+retry:
+ pushw %cx /* push the retry count on the stack. */
+
+ movw sec, %cx /* get sector number */
+ movw head, %dx /* get head number */
+ movb drive, %dl
+
+ movw $BD_READ1, %ax /* read 1 sector */
+ int $BIOS_DISK
+ jnc readcont /* jmp if it worked ok */
+
+ movw $err_diskread, %si /* report the error */
+ call dispMsg
+ popw %cx /* get the retry count back */
+ loop retry /* and go try again. */
+ movw $err_diskfail, %si /* can't proceed, */
+ call dispMsg /* print message and freeze. */
+ jmp .
+
+readcont:
+ movw $msg_dot, %si /* print status: a dot */
+ call dispMsg
+ cmpw $OFFSET_LIMIT, %bx /* have we reached the offset limit? */
+ je adjust /* Yes--must adjust the es register */
+ addw $SECTOR_SIZE, %bx /* No--just adjust the block size to */
+ jmp readcont2 /* the offset and continue. */
+
+adjust:
+ movw $0, %bx /* start offset over again */
+ movw %es, %ax
+ addw $0x1000,%ax /* move segment pointer to next chunk */
+ movw %ax, %es
+
+readcont2:
+ incb %cl /* not done - move to the next sector */
+ cmpb max_sec, %cl /* see if we need */
+ jnz save_sector /* to switch heads or tracks */
+
+ movb $1, %cl /* reset sector number */
+ incb %dh /* first, switch heads */
+ cmpb max_head, %dh /* there are only two - if we've already */
+ jnz save_sector /* used both, we need to switch tracks */
+
+ xorb %dh, %dh /* reset to head 0 */
+ incb %ch /* inc track number */
+ cmpb $80, %ch /* 80 tracks per side - have we read all? */
+ jnz save_sector /* read another track */
+
+ movw $err_toobig, %si /* report the error */
+ call dispMsg
+ jmp . /* and freeze */
+
+save_sector:
+ movw %cx, sec /* save sector number */
+ movw %dx, head /* and head number */
+
+ popw %ax /* discard the retry count */
+ popw %ax /* get the sector count from the stack */
+ decw %ax /* and decrement it. */
+ jg readprog /* If it is zero, we're done reading. */
+
+readdone:
+ movw $msg_bar, %si /* print message saying this block is done */
+ call dispMsg
+ ret /* and return to the caller */
+
+/*
+** We've loaded the whole target program into memory,
+** so it's time to transfer to the startup code.
+*/
+done_loading:
+ movw $msg_go, %si /* last status message */
+ call dispMsg
+
+ jmp switch /* move to the next phase */
+
+/*
+** Support routine - display a message byte by byte to the monitor.
+*/
+dispMsg:
+ pushw %ax
+ pushw %bx
+repeat:
+ lodsb /* grab next character */
+
+ movb $BV_W_ADV, %ah /* write and advance cursor */
+ movw $0x07, %bx /* page 0, white on blank, no blink */
+ orb %al, %al /* AL is character to write */
+ jz getOut /* if we've reached the NUL, get out */
+
+ int $BIOS_VIDEO /* otherwise, print and repeat */
+ jmp repeat
+
+getOut: /* we're done, so return */
+ popw %bx
+ popw %ax
+ ret
+
+/*
+** Support routine - move the GDT entries from where they are to
+** location 0050:0000. We need to add BOOT_ADDR because the bootstrap
+** is linked at 0, but loaded at 0x7c00.
+*/
+move_gdt:
+ movw %cs, %si
+ movw %si, %ds
+ movw $start_gdt + BOOT_ADDR, %si
+ movw $GDT_SEG, %di
+ movw %di, %es
+ xorw %di, %di
+ movl $gdt_len, %ecx
+ cld
+ rep movsb
+ ret
+
+/*
+** DATA AREAS.
+**
+** Next sector number and head number to read from.
+*/
+sec: .word 2 /* cylinder=0, sector=1 */
+head: .word 0 /* head=0 */
+max_sec: .byte 19 /* up to 18 sectors per floppy track */
+max_head: .byte 2 /* only two r/w heads per floppy drive */
+
+/*
+** Status and error messages.
+*/
+msg_loading: .asciz "Loading"
+msg_dot: .asciz "."
+msg_go: .asciz "done."
+msg_bar: .asciz "|"
+
+/*
+** Error messages.
+*/
+err_diskstatus: .asciz "Disk not ready.\n\r"
+err_diskread: .asciz "Read failed\n\r"
+err_toobig: .asciz "Too big\n\r"
+err_diskfail: .asciz "Can't proceed\n\r"
+
+/*
+** Data areas.
+*/
+
+/*
+** The GDTR and IDTR contents.
+*/
+gdt_48:
+ .word 0x2000 /* 1024 GDT entries x 8 bytes/entry = 8192 */
+ .quad GDT_ADDR
+
+idt_48:
+ .word 0x0800 /* 256 interrupts */
+ .quad IDT_ADDR
+
+/*
+** Depending on the age of the BIOS, it may expect there to be a
+** partition table for the hard drive you're booting from at this point
+** in the boot sector; only the first 446 bytes (0x000-0x1bd) can be
+** used for bootstrap code/data. To make life easy, we'll just skip
+** over the rest of the sector.
+**
+** Note: when booting from floppy, this isn't a problem, because floppy
+** disks don't have partition tables. On some machines, USB-type storage
+** devices are treated as floppies, so they also don't have partition
+** maps; however, on other systems, USB storage is treated as hard disk
+** storage.
+*/
+
+/*
+** End of the first sector of the boot program. The last two bytes
+** of this sector must be AA55 in order for the disk to be recognized
+** by the BIOS as bootable.
+*/
+ .org SECTOR_SIZE-4
+
+drive: .word BDEV /* 0x00 = floppy, 0x80 = usb */
+
+boot_sig:
+ .word 0xAA55
+
+/*******************************************************
+******* BEGINNING OF SECTOR TWO OF THE BOOTSTRAP *******
+*******************************************************/
+
+/*
+** This code configures the GDT, enters protected mode, and then
+** transfers to the OS entry point.
+*/
+
+switch:
+ cli
+ movb $NMI_DISABLE, %al /* also disable NMIs */
+ outb %al, $CMOS_ADDR
+
+#ifdef USE_FLOPPY
+ call floppy_off
+#endif
+ call enable_A20
+ call move_gdt
+#ifdef GET_MMAP
+ call check_memory
+#endif
+
+/*
+** Get the memory address for the "user blob" out of the table
+** at the end of this sector, and pass the three values to the
+** protected mode code in %bx, %cx, and %dx. We could figure out
+** how to find it from there, but this is easier.
+*/
+# movw u_off+BOOT_ADDR, %bx
+# movw u_seg+BOOT_ADDR, %cx
+# movw u_sect+BOOT_ADDR, %dx
+
+/*
+** The IDTR and GDTR are loaded relative to this segment, so we must
+** use the full offsets from the beginning of the segment (0000:0000);
+** however, we were loaded at 0000:7c00, so we need to add that in.
+*/
+ lidt idt_48 + BOOT_ADDR
+ lgdt gdt_48 + BOOT_ADDR
+
+ movl %cr0, %eax /* get current CR0 */
+ orl $1, %eax /* set the PE bit */
+ movl %eax, %cr0 /* and store it back. */
+
+ /*
+ ** We'll be in protected mode at the start of the user's code
+ ** right after this jump executes.
+ **
+ ** First, a byte to force 32-bit mode execution, followed by
+ ** a 32-bit long jump. The long ("far") jump loads both EIP
+ ** and CS with the proper values so that when we land at the
+ ** destination address in protected mode, the next instruction
+ ** fetch doesn't cause a fault.
+ **
+ ** The old code for this:
+ **
+ ** .byte 0x66, 0xEA
+ ** .long TARGET_ADDR
+ ** .word GDT_CODE
+ */
+
+ .byte 0x66
+ .code32
+ ljmp $GDT_CODE, $TARGET_ADDR
+ .code16
+
+/*
+** Supporting functions.
+*/
+
+#ifdef USE_FLOPPY
+/*
+** Turn off the motor on the floppy disk drive.
+*/
+floppy_off:
+ push %dx
+ movw $0x3f2, %dx
+ xorb %al, %al
+ outb %al, %dx
+ pop %dx
+ ret
+#endif
+
+/*
+** Enable the A20 gate for full memory access.
+*/
+enable_A20:
+ call a20wait
+ movb $KBD_P1_DISABLE, %al
+ outb %al, $KBD_CMD
+
+ call a20wait
+ movb $KBD_RD_OPORT, %al
+ outb %al, $KBD_CMD
+
+ call a20wait2
+ inb $KBD_DATA, %al
+ pushl %eax
+
+ call a20wait
+ movb $KBD_WT_OPORT, %al
+ outb %al, $KBD_CMD
+
+ call a20wait
+ popl %eax
+ orb $2, %al
+ outb %al, $KBD_DATA
+
+ call a20wait
+ mov $KBD_P1_ENABLE, %al
+ out %al, $KBD_CMD
+
+ call a20wait
+ ret
+
+a20wait: /* wait until bit 1 of the device register is clear */
+ movl $65536, %ecx /* loop a lot if need be */
+wait_loop:
+ inb $KBD_STAT, %al /* grab the byte */
+ test $2, %al /* is the bit clear? */
+ jz wait_exit /* yes */
+ loop wait_loop /* no, so loop */
+ jmp a20wait /* if still not clear, go again */
+wait_exit:
+ ret
+
+a20wait2: /* like a20wait, but waits until bit 0 is set. */
+ mov $65536, %ecx
+wait2_loop:
+ in $KBD_STAT, %al
+ test $1, %al
+ jnz wait2_exit
+ loop wait2_loop
+ jmp a20wait2
+wait2_exit:
+ ret
+
+#ifdef GET_MMAP
+/*
+** Query the BIOS to get the list of usable memory regions
+**
+** Adapted from: http://wiki.osdev.org/Detecting_Memory_%28x86%29
+** (see section "BIOS Function INT 0x15. EAX = 0xE820")
+**
+** After the first 'int', if the location 0x2D00 (4 bytes) contains -1,
+** then this method failed to detect memory properly; otherwise, this
+** location contains the number of elements read.
+**
+** The start of the array is at 0x2D04. The elements are tightly
+** packed following the layout as defined below. Each entry in the
+** array contains the following information:
+**
+** uint64_t base address of region
+** uint64_t length of region (0 --> ignore the entry)
+** uint32_t type of region
+** uint32_t ACIP 3.0 Extended Attributes
+**
+** The C struct definition is as follows:
+**
+** struct MemMapEntry
+** {
+** uint32_t base[2]; // 64-bit base address
+** uint32_t length[2]; // 64-bit length
+** uint32_t type; // 32-bit region type
+** uint32_t ACPI; // 32-bit ACPI "extended attributes" bitfield
+** };
+**
+** This structure must be packed in memory. This shouldn't be a problem,
+** but if it is, you may need to add this attribute at the end of the
+** struct declaration before the semicolon:
+**
+** __attribute__((packed))
+**
+** Parameters:
+** None
+**/
+check_memory:
+ // save everything
+ // pushaw won't work here because we're still in real mode
+ pushw %ds
+ pushw %es
+ pushw %ax
+ pushw %bx
+ pushw %cx
+ pushw %dx
+ pushw %si
+ pushw %di
+
+ // Set the start of the buffer
+ movw $MMAP_SEG, %bx // 0x2D0
+ mov %bx, %ds // Data segment now starts at 0x2D00
+ mov %bx, %es // Extended segment also starts at 0x2D00
+
+ // Reserve the first 4 bytes for the # of entries
+ movw $0x4, %di
+ // Make a valid ACPI 3.X entry
+ movw $1, %es:20(%di)
+
+ xorw %bp, %bp // Count of entries in the list
+ xorl %ebx, %ebx // EBX must contain zeroes
+
+ movl $MMAP_MAGIC_NUM, %edx // Magic number into EDX
+ movl $MMAP_CODE, %eax // E820 memory command
+ movl $MMAP_ENT, %ecx // Ask the BIOS for 24 bytes
+ int $BIOS_MISC // Call the BIOS
+
+ // check for success
+ jc cm_failed // C == 1 --> failure
+ movl $MMAP_MAGIC_NUM, %edx // sometimes EDX changes
+ cmpl %eax, %edx // EAX should equal EDX after the call
+ jne cm_failed
+ testl %ebx, %ebx // Should have at least one more entry
+ je cm_failed
+
+ jmp cm_jumpin // Good to go - start us off
+
+cm_loop:
+ movl $MMAP_CODE, %eax // Reset our registers
+ movw $1, 20(%di)
+ movl $MMAP_ENT, %ecx
+ int $BIOS_MISC
+ jc cm_end_of_list // C == 1 --> end of list
+ movl $MMAP_MAGIC_NUM, %edx
+
+cm_jumpin:
+ jcxz cm_skip_entry // Did we get any data?
+
+ cmp $20, %cl // Check the byte count
+ jbe cm_no_text // Skip the next test if only 20 bytes
+
+ testb $1, %es:20(%di) // Check the "ignore this entry" flag
+ je cm_skip_entry
+
+cm_no_text:
+ mov %es:8(%di), %ecx // lower half of length
+ or %es:12(%di), %ecx // now, full length
+ jz cm_skip_entry
+
+ inc %bp // one more valid entry
+
+ // make sure we don't overflow our space
+ cmpw $MMAP_MAX_ENTS, %bp
+ jge cm_end_of_list
+
+ // we're ok - move the pointer to the next struct in the array
+ add $24, %di
+
+cm_skip_entry:
+ // are there more entries to retrieve?
+ testl %ebx, %ebx
+ jne cm_loop
+
+cm_end_of_list:
+ // All done! Store the number of elements in 0x2D00
+ movw %bp, %ds:0x0
+
+ clc // Clear the carry bit and return
+ jmp cm_ret
+
+cm_failed:
+ movl $-1, %ds:0x0 // indicate failure
+ stc
+
+cm_ret:
+ // restore everything we saved
+ // popaw won't work here (still in real mode!)
+ popw %di
+ popw %si
+ popw %dx
+ popw %cx
+ popw %bx
+ popw %ax
+ popw %es
+ popw %ds
+ ret
+#endif
+
+/*
+** The GDT. This cannot be created in C because the bootstrap is not
+** linked with that code. We could just have a simple "dummy" GDT here
+** but that would only save us a couple of entries. Also, we could save
+** some space by not having the separate 'linear' and 'stack' entries
+** (they're identical to the 'data' entry).
+*/
+ .p2align 2 // force 4-byte alignment
+start_gdt:
+ // selector 0x0000 is unused
+ SEGNULL
+
+ // selector 0x0008 - basic linear access to all of memory
+ SEGMENT( 0x0, 0xffffffff, SEG_DPL_0, SEG_DATA_RW )
+
+ // selector 0x0010 - kernel code segment
+ SEGMENT( 0x0, 0xffffffff, SEG_DPL_0, SEG_CODE_XR )
+
+ // selector 0x0018 - kernel data segment
+ SEGMENT( 0x0, 0xffffffff, SEG_DPL_0, SEG_DATA_RW )
+
+ // selector 0x0020 - kernel stack segment
+ SEGMENT( 0x0, 0xffffffff, SEG_DPL_0, SEG_DATA_RW )
+
+ // could put additional entries here for user mode - e.g.,
+ // 0x0028 code: SEGMENT( 0x0, 0xffffffff, SEG_DPL_3, SEG_CODE_XR )
+ // 0x0030 data: SEGMENT( 0x0, 0xffffffff, SEG_DPL_3, SEG_DATA_RW )
+ // 0x0038 stack: SEGMENT( 0x0, 0xffffffff, SEG_DPL_3, SEG_DATA_RW )
+
+end_gdt:
+gdt_len = end_gdt - start_gdt
+
+/*
+** The end of this program will contain a list of the sizes and load
+** addresses of all of the blocks to be loaded. These values are
+** inserted here by the BuildImage program, which checks that there are
+** not so many blocks that the GDT would be overwritten. The layout
+** of the data is:
+**
+** struct info_s {
+** short offset;
+** short segment;
+** short sectors;
+** };
+**
+** with the data for the first program at k_off, k_seg, and k_sect.
+** If additional blocks are to be loaded, their values appear just
+** before the previous set.
+*/
+
+ .org BOOT_SIZE-12
+u_off: .word 0 // the "user blob"
+u_seg: .word 0
+u_sect: .word 0
+k_off: .word 0 // the kernel
+k_seg: .word 0
+k_sect: .word 0