使用grub引导程序集编写内核

时间:2014-04-01 16:15:52

标签: assembly boot osdev grub

我知道使用一个简单的内核这个小实际上我不需要用GRUB启动它但是我试图在我的内核变大之前学习如何做到这一点我需要启动它。我在OSDev.org上使用实模式汇编教程编写了一个带汇编的简单内核。我尝试启动它,但GRUB给了我一个错误,它无法找到多启动头。我看了一下C Bare Bones教程的汇编代码。我是否只需将所有内容放在.multiboot中?你能告诉我如何启动这段代码吗? (这段代码不属于我的创作,我已将代码留在家用机器上,现在我在学校,我只是从OSDev借用了这段代码,所以请不要给我任何平均评论,说我偷了某人&# 39; s代码。)

; boot.asm
mov ax, 0x07c0
mov ds, ax

mov si, msg
ch_loop:lodsb
or al, al ; zero=end or str
jz hang   ; get out
mov ah, 0x0E
int 0x10
jmp ch_loop

hang:
jmp hang

msg   db 'Welcome to Macintosh', 13, 10, 0
times 510-($-$$) db 0
db 0x55
db 0xAA

我可以使用BIOS启动和打印,还是使用堆栈?

;====================================

[ORG 0x7c00]      ; add to offsets
xor ax, ax    ; make it zero
mov ds, ax   ; DS=0
mov ss, ax   ; stack starts at 0
mov sp, 0x9c00   ; 200h past code start

mov ax, 0xb800   ; text video memory
mov es, ax

mov si, msg   ; show text string
call sprint

mov ax, 0xb800   ; look at video mem
mov gs, ax
mov bx, 0x0000   ; 'W'=57 attrib=0F
mov ax, [gs:bx]

mov  word [reg16], ax ;look at register
call printreg16

hang:
jmp hang

----------------------
dochar:   call cprint         ; print one character
sprint:   lodsb      ; string char to AL
cmp al, 0
jne dochar   ; else, we're done
add byte [ypos], 1   ;down one row
mov byte [xpos], 0   ;back to left
ret

cprint:   mov ah, 0x0F   ; attrib = white on black
mov cx, ax    ; save char/attribute
movzx ax, byte [ypos]
mov dx, 160   ; 2 bytes (char/attrib)
mul dx      ; for 80 columns
movzx bx, byte [xpos]
shl bx, 1    ; times 2 to skip attrib

mov di, 0        ; start of video memory
add di, ax      ; add y offset
add di, bx      ; add x offset

mov ax, cx        ; restore char/attribute
stosw              ; write char/attribute
add byte [xpos], 1  ; advance to right

ret

;------------------------------------

printreg16:
mov di, outstr16
mov ax, [reg16]
mov si, hexstr
mov cx, 4   ;four places
hexloop:
rol ax, 4   ;leftmost will
mov bx, ax   ; become
and bx, 0x0f   ; rightmost
mov bl, [si + bx];index into hexstr
mov [di], bl
inc di
dec cx
jnz hexloop

mov si, outstr16
call sprint

ret

;------------------------------------

xpos   db 0
ypos   db 0
hexstr   db '0123456789ABCDEF'
outstr16   db '0000', 0  ;register value string
reg16   dw    0  ; pass values to printreg16
msg   db "What are you doing, Dave?", 0
times 510-($-$$) db 0
db 0x55
db 0xAA
;==================================

提前谢谢你。

2 个答案:

答案 0 :(得分:1)

如果要编写实模式内核(16位),则无法使用GRUB。 GRUB只能加载32位内核。

您应该考虑一下这样一个事实,即只能在32位模式下使用BIOS中断。

MultiBoot规范(http://www.uruk.org/orig-grub/boot-proposal.html)描述了GRUB所需的文件格式。

对于初学者来说绝对没有...

如果你想在不使用GRUB的情况下启动内核(并且你的操作系统大于510字节),你的启动扇区必须使用中断0x13从磁盘加载其余的操作系统。

此时计算机以16位实模式运行。

答案 1 :(得分:1)

使用多重引导,内核以受保护模式here启动。

  

'CR0'必须清零位31(PG)。必须设置位0(PE)。其他比特   都未定义。

Grub可以启动16位内核,因为Linux以16位here启动。但对于Linux,FreeBSD,NetBSD和OpenBSD来说,它是为了方便here。并here

  

不支持多重启动但没有支持的操作系统   GRUB中的特定支持(特定支持适用于Linux,   FreeBSD,NetBSD和OpenBSD)必须是链式加载的,这涉及到   加载另一个引导加载程序并以实模式跳转到它