Compare commits

...

2 Commits

3 changed files with 98 additions and 97 deletions

View File

@@ -140,15 +140,7 @@ main:
je main.sig_ok je main.sig_ok
ERROR MBR_ERROR_NO_VBR_SIG ; no signature present ERROR MBR_ERROR_NO_VBR_SIG ; no signature present
.sig_ok: .sig_ok:
push PartTable_t_size ; len
push DiskSig ; src -> start of partition table
push partition_table ; dst -> addr in bss
call kmemcpy ; copy partition table to bss
add sp, 0x6
mov si, word [bp - 4] ; partition_offset address
mov dl, byte [bp - 2] ; pass drive # from BIOS to VBR in dl mov dl, byte [bp - 2] ; pass drive # from BIOS to VBR in dl
mov bx, partition_table ; partition_table address
jmp word 0x0000:VBR_ENTRY jmp word 0x0000:VBR_ENTRY
; ############### ; ###############
@@ -181,14 +173,11 @@ BootSig:
section .bss follows=.text section .bss follows=.text
begin_bss: begin_bss:
align 16, resb 1
partition_table resb PartTable_t_size
align 16, resb 1 align 16, resb 1
lba_packet resb LBAPkt_t_size lba_packet resb LBAPkt_t_size
align 512, resb 1 align 512, resb 1
stack_bottom resb 512 - 16 ; 512 byte stack early on stack_bottom resb 1024 - 16 ; 512 byte stack early on
stack_top: stack_top:
mbr_redzone resb 16 mbr_redzone resb 16
end_bss: end_bss:

View File

@@ -36,18 +36,10 @@
section .text section .text
begin_text: begin_text:
; dl = byte boot_drive ; dl = byte boot_drive
; ax = word part_offset (active partition offset)
; si = ptr PartTable_t partition_table
; di = ptr FAT32_bpb_t fat32_bpb
ALIGN 4, db 0x90 ALIGN 4, db 0x90
init: init:
cli ; We do not want to be interrupted cli ; We do not want to be interrupted
; these 4 are stored in the .data section and are effectivly const types
mov [vbr_part_table_ptr], si ; pointer to partition_table
mov [vbr_fat32_bpb_ptr], di ; pointer to fat32_bpb
mov [boot_drive], dl ; copy boot_drive to globals mov [boot_drive], dl ; copy boot_drive to globals
mov [partition_offset], ax ; copy partition_offset to globals
mov ax, __STAGE2_SEGMENT ; set all our segments to the configured segment, except es mov ax, __STAGE2_SEGMENT ; set all our segments to the configured segment, except es
mov ds, ax ; * mov ds, ax ; *
@@ -129,22 +121,34 @@ main:
je main.stage2_main je main.stage2_main
ERROR STAGE2_SIGNATURE_MISSING ERROR STAGE2_SIGNATURE_MISSING
.stage2_main: .stage2_main:
; copy partition table data to .data section in stage2
__CDECL16_CALL_ARGS partition_table, word [vbr_part_table_ptr], PartTable_t_size
__CDECL16_CALL kmemcpy, 3
; copy bpb & ebpb to memory
__CDECL16_CALL_ARGS fat32_bpb, word [vbr_fat32_bpb_ptr], (FAT32_bpb_t_size + FAT32_ebpb_t_size)
__CDECL16_CALL kmemcpy, 3
call SetTextMode call SetTextMode
call disable_cursor_bios call disable_cursor_bios
; setup the early heap
__CDECL16_CALL_ARGS early_heap_state
__CDECL16_CALL arena_init, 1
__CDECL16_CALL_ARGS HelloPrompt_info __CDECL16_CALL_ARGS HelloPrompt_info
__CDECL16_CALL PrintString, 1 __CDECL16_CALL PrintString, 1
; setup the early heap ; setup and store our vbr/mbr (e)bpb
__CDECL16_CALL_ARGS early_heap_state __CDECL16_CALL_ARGS 0x200, 0x10
__CDECL16_CALL arena_alloc, 2
mov word [mbr_ptr], ax
push ax ; dst
movzx ax, byte [boot_drive]
push ax ; boot_drive
__CDECL16_CALL read_mbr, 2 ; fill mbr buffer
__CDECL16_CALL_ARGS 0x200, 0x10
__CDECL16_CALL arena_alloc, 2
mov word [vbr_ptr], ax
push ax ; dst
movzx ax, byte [boot_drive]
push ax ; boot_drive
__CDECL16_CALL read_vbr, 2 ; fill vbr buffer
; enable A20 gate ; enable A20 gate
call EnableA20 call EnableA20
@@ -197,55 +201,80 @@ hcf:
; destination buffer needs 512 bytes of space ; destination buffer needs 512 bytes of space
read_mbr: read_mbr:
__CDECL16_PROC_ENTRY __CDECL16_PROC_ENTRY
.func: .proc:
__BOCHS_MAGIC_DEBUG
; read mbr on boot drive to memory (for the partition table) ; read mbr on boot drive to memory (for the partition table)
movzx ax, byte [bp - 4] movzx ax, byte [bp + 4]
push ax ; drive_num (2) push ax ; drive_num (2)
push 0x01 ; count (2) push 0x01 ; count (2)
push dword 0x0 ; lba (4) push dword 0x0 ; lba (4)
push word [bp - 6] ; offset = dst push word [bp + 6] ; offset = dst
push __STAGE2_SEGMENT ; this segment push __STAGE2_SEGMENT ; this segment
__CDECL16_CALL read_disk_raw, 12 call read_disk_raw
add sp, 0xC
.check_sig:
mov bx, [bp + 6]
cmp word [bx + 0x1FE], 0xAA55 ; check for bytes at end
jne read_mbr.error
; TODO: this needs error checking, zero checking, check the sig a bunch of stuff... ; TODO: this needs error checking, zero checking, check the sig a bunch of stuff...
.endf: .update_globals:
mov ax, [bp + 6]
add ax, 0x1B8 ; offset to start of PartTable
mov [part_table_ptr], ax
.endp:
__CDECL16_PROC_EXIT __CDECL16_PROC_EXIT
ret
.error: .error:
ERROR STEVIA_DEBUG_ERR ERROR STEVIA_DEBUG_ERR
; int read_vbr(int boot_drive, void* mbr, void* buf) ; int read_vbr(int boot_drive, void* buf)
read_vbr: read_vbr:
__CDECL16_PROC_ENTRY __CDECL16_PROC_ENTRY
.func: .proc:
__BOCHS_MAGIC_DEBUG __BOCHS_MAGIC_DEBUG
; read vbr on boot partition to memory (for fat bpb/ebpb) ; read vbr on boot partition to memory (for fat bpb/ebpb)
.calc_part_offset: .calc_part_offset:
mov bx, [bp - 6] mov bx, word [part_table_ptr] ; base pointer @ partition table
add bx, 0x1B8 ; offset to partition table in mbr
mov cx, 4 ; only checking 4 entries mov cx, 4 ; only checking 4 entries
mov si, PartEntry_t.attributes
.find_active_L0: .find_active_L0:
mov al, byte [bx + PartEntry_t.attributes] mov al, byte [bx + si]
test al, 0x80 ; 0x80 == 1000_0000b test al, 0x80 ; 0x80 == 1000_0000b
jnz read_vbr.active_found je read_vbr.active_found
add bx, 0x10 ; add 16 bytes to offset add si, 0x10 ; add 16 bytes to offset (next part entry's attributes)
loop read_vbr.find_active_L0 loop read_vbr.find_active_L0
jmp read_vbr.error jmp read_vbr.error
.active_found: .active_found:
movzx ax, byte [bp - 4] add bx, si ; update base to active part
movzx ax, byte [bp + 4]
push ax ; drive_num (2) push ax ; drive_num (2)
push 0x01 ; count (2) push 0x01 ; count (2)
mov eax, dword [bx + PartEntry_t.lba_start] mov eax, dword [bx + PartEntry_t.lba_start]
push dword eax ; lba (4) push dword eax ; lba (4)
push word [bp - 6] ; offset = dst push word [bp + 6] ; offset = dst
push __STAGE2_SEGMENT ; this segment push __STAGE2_SEGMENT ; this segment
__CDECL16_CALL read_disk_raw, 12 call read_disk_raw
add sp, 0xC
; vbr (with fat bpb/ebpb) is at the buffer now ; vbr (with fat bpb/ebpb) is at the buffer now
.endf: .check_sig:
mov bx, [bp + 6]
cmp word [bx + 0x1FE], 0xAA55 ; check for bytes at end
jne read_mbr.error
.check_FAT_size:
test word [bx + FAT32_bpb_t.unused2_ZERO_word], 0 ; TotSectors16 will not be set if FAT32
jnz read_vbr.error
.update_globals:
mov word [fat32_bpb_ptr], bx
add bx, FAT32_bpb_t_size ; offset to ebpb
mov word [fat32_ebpb_ptr], bx
.endp:
__CDECL16_PROC_EXIT __CDECL16_PROC_EXIT
ret
.error: .error:
ERROR STEVIA_DEBUG_ERR ERROR STEVIA_DEBUG_ERR
@@ -386,25 +415,14 @@ BootTarget:
db 'BOOT BIN' db 'BOOT BIN'
; ;
; pre-bss init globals (generally const...but there are exceptions) ; pre-bss init globals
; ;
align 8, db 0 ; set to boot_drive passed from BIOS almost first thing in init
align 4, db 0
boot_drive: boot_drive:
db 0x00 db 0x00
align 8, db 0
partition_offset:
dw 0x0000
align 8, db 0
vbr_fat32_bpb_ptr:
dw 0x0000
align 8, db 0
vbr_part_table_ptr:
dw 0x0000
align 16, db 0 align 16, db 0
IntToHex_table: IntToHex_table:
db '0123456789ABCDEF' db '0123456789ABCDEF'
@@ -497,8 +515,28 @@ STAGE2_SIG: dd 0xDEADBEEF ; Signature to mark the end of the stag
section .bss follows=.sign section .bss follows=.sign
begin_bss: begin_bss:
; structures
align 4, resb 1
mbr_ptr:
resw 1
align 4, resb 1
vbr_ptr:
resw 1
align 4, resb 1
part_table_ptr:
resw 1
align 4, resb 1
fat32_bpb_ptr:
resw 1
align 4, resb 1
fat32_ebpb_ptr:
resw 1
; structures
align 16, resb 1 align 16, resb 1
partition_table: partition_table:
resb PartTable_t_size resb PartTable_t_size

View File

@@ -21,6 +21,7 @@
%define __STEVIA_VBR %define __STEVIA_VBR
section .text section .text
__ENTRY: __ENTRY:
phy_bpb_start:
jmp short (init - $$) jmp short (init - $$)
nop nop
@@ -92,24 +93,8 @@ init:
ALIGN 4, db 0x90 ALIGN 4, db 0x90
main: main:
mov byte [bp - 2], dl ; boot_drive mov byte [bp - 2], dl ; boot_drive
mov word [bp - 4], si ; part_offset (i.e offset into partition table our boot partition is
mov word [bp - 6], bx ; partition_table
.load_fs_data:
push PartTable_t_size ; count=
push word [bp - 6] ; src= ptr partition_table
push partition_table ; dst=
call kmemcpy ; copy partition table data to bss
add sp, 0x6
push (FAT32_bpb_t_size + FAT32_ebpb_t_size) ; size in byte, should be 90 bytes
push __ENTRY ; src
push fat32_bpb ; dst
call kmemcpy ; copy bpb & ebpb to bss
add sp, 0x6
.check_FAT_size: ; we only support a very specific setup of FAT32 .check_FAT_size: ; we only support a very specific setup of FAT32
mov bx, fat32_bpb mov bx, phy_bpb_start
test word [bx + FAT32_bpb_t.unused2_ZERO_word], 0 ; TotSectors16 will not be set if FAT32 test word [bx + FAT32_bpb_t.unused2_ZERO_word], 0 ; TotSectors16 will not be set if FAT32
jz main.load_stage2 jz main.load_stage2
ERROR VBR_ERROR_WRONG_FAT_SIZE ERROR VBR_ERROR_WRONG_FAT_SIZE
@@ -132,11 +117,7 @@ main:
call read_disk_raw call read_disk_raw
add sp, 0xC add sp, 0xC
.enter_stage2: .enter_stage2:
; TODO: review what we pass to stage2, do we need to do this?
mov dl, byte [bp - 2] ; byte boot_drive mov dl, byte [bp - 2] ; byte boot_drive
mov ax, word [bp - 4] ; word part_offset
mov si, partition_table ; ptr partition_table
mov di, fat32_bpb ; ptr fat32_bpb
jmp word 0x0000:STAGE2_ENTRY jmp word 0x0000:STAGE2_ENTRY
; ############### ; ###############
@@ -157,18 +138,11 @@ BootSig:
section .bss follows=.text section .bss follows=.text
begin_bss: begin_bss:
align 16, resb 1
partition_table resb PartTable_t_size
align 16, resb 1
fat32_bpb resb FAT32_bpb_t_size
fat32_ebpb resb FAT32_ebpb_t_size
align 16, resb 1 align 16, resb 1
lba_packet resb LBAPkt_t_size lba_packet resb LBAPkt_t_size
align 512, resb 1 align 512, resb 1
stack_bottom resb (512 - 16) ; 512b stack early on stack_bottom resb (1024 - 16) ; 512b stack early on
stack_top: stack_top:
vbr_redzone resb 16 vbr_redzone resb 16
end_bss: end_bss: