System模块在磁盘上的位置紧接着setup.s所在扇区,即0磁道0磁头3扇区。该模块被读到内存地址0x10000开始处,模块长度0x30000。读入该模块的代码被封装到了read_it函数中。
由于实模式下段偏移寄存器只有16位,最多寻址0x10000内的地址,所以无法在不改变段寄存器的情况下读入整个system模块。程序选择的方法是每次读入0x10000,然后将段寄存器加上这个偏移,反复执行直到读完。
在一次大小为0x10000读取中,程序要比较剩余需要读取的扇区数a和当前磁道、磁头下剩余可读扇区数b的大小。如果a>b,则读取b个扇区;否则读取a个扇区。一次读取完成后,要重新设置当前磁道号、磁头号和扇区号,以及内存缓冲区位置,包括段寄存器和段偏移。
循环过程中各关键变量的名字如下:track=当前磁道号,head=当前磁头号,sread=当前已读扇区数,ES:bx=缓冲区头部位置,al=一次读取中需要读取的扇区数。
函数的关键流程图如下:
附上汇编代码:
read_it: !在进入函数之前es被初始化为0x1000
mov ax,es
test ax,#0x0fff !测试es低12位是否为0,实模式下ES还有四位隐藏位。实际测试低16位是否为0,即测试es是否位于64kb边界。test指令对两个操作数位与,若结果为0则ZF标志位置位。
die: jne die ! es must be at 64kB boundary 若ZF=0,死机;否则初始化段偏移bx=0
xor bx,bx ! bx is starting address within segment
rp_read: !开始进入读循环,循环终止条件为es=ENDSEG(0x4000)
mov ax,es
cmp ax,#ENDSEG ! have we loaded a