1.用Bochs-2.4.6进行调试
.bxrc文件做成
---------------------------------------------------------------
megs:4romimage: file=../BIOS-bochs-latestvgaromimage: file=../VGABIOS-lgpl-latestvga: extension=vbefloppya: 1_44=mytest.img,status=insertedboot: alog: bochsout.txtmouse: enabled=0vga_update_interval: 150000
---------------------------------------------------------------
执行以下命令
"D:\Program Files (x86)\Bochs-2.4.6\bochsdbg.exe" -q -f boot.bxrc
基本调试命令:
下断点 b addr //b 0x7c00
断点列表 blist
执行到断点 c
单步步入 s
单步步过 n
寄存器信息 info cpu r fp sreg(查看cs) creg
查看内存(物理) xp /nuf addr //xp /40bc 0x90000 f=format :x d u o t c
反汇编内存 u start end
2.
现象:第一扇区被加载后,自己实现的扇区载入代码,在载入扇区内容后,字符串无法显示
原代码:
!代码段
mov cx,#17
mov bx,#0x0007
mov bp,#msg3
sub bp,#l1
mov ax,#0x1301
int 0x10
msg3:.byte 13,10.ascii "finished...".byte 13,10,13,10
原因:扇区内容被载入到自己指定的位置,导致与原来编译器指定的地址不同,所以中断显示字符串失败。
修正方法:根据载入内存后代码直接的相对偏移不变,进行载入后重定位,计算出正确的地址。
tips:
1. 编译时追加产生中间文件,如list文件
as86 -0 -a -o -l list.s boot.o boot.s
修改后
!代码段
mov cx,#17
mov bx,#0x0007
call l1
l1:
pop bp
add bp,#msg3
sub bp,#l1
mov ax,#0x1301
int 0x10
msg3:.byte 13,10.ascii "finished...".byte 13,10,13,10
3.
现象:boot载入后、屏幕循环打印同一字符串
原因:代码错误
代码:
read_track:
push ax
push bx
push cx
push dx
!int 0x13将后续模块从第2个扇区开始读入
!到0x10000开始
!INT 0x13 的使用方法如下:
! 读扇区:
! ah = 0x02 - 读磁盘扇区到内存;al = 需要读出的扇区数量;
! ch = 磁道(柱面)号的低8 位; cl = 开始扇区(0-5 位),磁道号高2 位(6-7);
! dh = 磁头号; dl = 驱动器号(如果是硬盘则要置位7);
! es:bx ??指向数据缓冲区; 如果出错则CF 标志置位。
mov dx,track
mov cx,sread
inc cx !cl=开始读扇区
!//---------------------------------这行漏了mov ch,dl,抄代码不容易啊
mov dx,head
mov dh,dl
mov dl,#0
and dx,#0x0100
mov ah,#2
int 0x13
jc bad_rt
pop dx
pop cx
pop bx
pop ax
ret
bad_rt: mov ax,#0
mov dx,#0
int 0x13
pop dx
pop cx
pop bx
pop ax
jmp read_track
因为遗漏了mov ch,dl磁道号的更新,导致每次调用中断读扇区时,都从磁道号0开始,因此,在每次读入1个磁道后,又重新开始读取第0号的内容。
具体表现为:
第一次载入内存0x10000后,在读完一个磁道后,在地址0x14600处(17*512+18*512=0x4600,第1个扇区已有bios载入,1.44MB软盘)内容与0x10000相同(之后为0x14600+0x4800*n)。而0x10000处最后没有加死循环代码,导致相同的代码重复被执行。