参照《linux完全剖析》一文中的程序自己写了
如下两个简单的程序:
boot.s
;这个程序将软驱的第二个扇区中的内容(kernel)读到
;内存0x1000:0000的位置然后跳转到该处执行
;
;
;
org 07c00h
jmp start
msg:
db "Now,begin to load kernel....",0ah
msgLen equ $-msg
start:
mov ax,cs
mov ds,ax
mov es,ax
mov ax,msg
mov bp,ax
mov cx,msgLen
mov ax,01301h
mov bx,000ch
mov dl,0
int 10h
load:
mov dx,0x0000 ;dh->磁头号,dl->驱动器号
mov ax,0x1000
mov es,ax
xor bx,bx ;es:bx->the data 0x1000:0000
mov ch,0 ;ch->磁道号
mov cl,2 ;cl->起始扇区号
mov ah,02h ;param
mov al,01h ;how many blocks to read
int 13h
sys: ;jmp to the kernel
jmp 0x1000:0
times 510 -($-$$) db 0
dw 0xaa55
2、sys.s
mov ax,cs
mov ds,ax
mov es,ax
mov ss,ax
mov ax,msg
mov bp,ax
mov cx,msgLen
mov ax,01301h
mov bx,000ch
mov dl,0
int 10h
jmp $
msg:
db "It's the world of kernel!",0ah
msgLen equ $-msg
times 512-($-$$) db 0
将上述两个汇编程序编译之后生成boot.bin sys.bin 然后使用命令dd if=sys.bin of=boot.bin bs=512 seek=1 count=1 将这两个512字节的bin文件做成一个1024字节的文件(即boot在软盘的第一个扇区,sys在软盘的第二个扇区)
使用qemu-system-i386 -hda boot.bin 只出现了“Now,begin to load kernel....”一句话,换言之,sys并没有执行,但是使用bochs模拟却完全没有问题。
原因:应该使用qemu-system-i386 -fda boot.bin ,因为boot中使用的int 13h按照软驱的方式读取的,所以应该使用参数为-fda
若要使用硬盘启动的话,将程序的中的
load:
mov dx,0x0000
;dh->磁头号,dl->驱动器号
改为:
load:
mov dx,0x0080
;dh->磁头号,dl->驱动器号,软驱从0开始: ==> 软驱A:0 软驱B:1
;硬驱从80h开始: ===> 硬盘C:80h 硬盘D: 81h ....
即可!