中国DOS联盟论坛

中国DOS联盟

-- 联合DOS 推动DOS 发展DOS --

联盟域名:www.cn-dos.net  论坛域名:www.cn-dos.net/forum
DOS,代表着自由开放与发展,我们努力起来,学习FreeDOS和Linux的自由开放与GNU精神,共同创造和发展美好的自由与GNU GPL世界吧!

游客:  注册 | 登录 | 命令行 | 会员 | 搜索 | 上传 | 帮助 »
作者:
标题: DOS引导扇区代码详解[转帖] 取消高亮 | 上一主题 | 下一主题
ko20010214
版主




积分 7294
发帖 1628
注册 2002-10-16
状态 离线
『楼 主』:  DOS引导扇区代码详解[转帖]

如果从软盘起动,则Dos引导程序被ROM BIOS直接加载到内存,若从硬盘起动,则被硬
盘的主引导程序加载.不过都是被加载到内存的绝对地址0000:7C00H处.因此,Dos引导程
序的第一条指令的地址一定是0000:7C00H.

Dos引导程序所做的事情如下:
1>调整堆栈位置
2>修改并用修改后的磁盘参数表来复位磁盘系统
3>计算根目录表的首扇区的位置及IO.SYS的扇区位置
4>读入根目录表的首扇区
5>检查根目录表的开头两项是否为IO.SYS及MSDOS.SYS
6>将IO.SYS文件开头三个扇区读入内存0000:0700H处
7>跳到0000:0700H处执行IO.SYS,引导完毕
    上述每一步若出错,则显示"Non system disk or disk error..."信息,等用户按任
一键后试图重新起动.

    下面的Dos引导程序是从硬盘上得来的,显示MSDOS5.0,但Dos的ver命令报告的是6.22
版.FAT表自然是16位的.

说明:
    (DX)          表示寄存器DX的值
    逻辑扇区号    以0面0道1扇区作为逻辑0扇区,而不是以Dos引导扇区为逻辑0扇区,
                  当然,对软盘来说二者是相同的,对硬盘则不同
    面号          即磁头号
    磁道号        即柱面号(对硬盘)
    物理扇区号    由面号,磁道号,扇区号三者共同指定
偏移  机器码        符号指令               说明
============================================================================
==
0000 EB3C          JMP     003E            ;跳过数据区
                                           ;以下数据是厂商OEM信息和磁盘BPB表

0000        90 4D 53 44 4F 53-35 2E 30 00 02 08 01 00     .MSDOS5.0.....
0010  02 00 02 00 00 F8 CC 00-3F 00 10 00 3F 00 00 00   ........?...?...
0020  F1 59 06 00 80 00 29 E3-0B 3F 26 53 4C 4D 20 20   .Y....)..?&SLM
0030  20 20 20 20 20 20 46 41-54 31 36 20 20 20               FAT16
----------------------------------------------------------------------------
--
003E FA            CLI
003F 33C0          XOR     AX,AX
0041 8ED0          MOV     SS,AX
0041 8ED0          MOV     SS,AX
0043 BC007C        MOV     SP,7C00         ; 初始化堆栈
0046 16            PUSH    SS
0047 07            POP     ES              ;(ES)=0000H
0048 BB7800        MOV     BX,0078         ;1EH 号中断向量的地址为0000:0078H

004B 36            SS:                     ;(SS)=0000H
004C C537          LDS     SI,[BX]         ;取1EH号中断向量的内容存入DS:SI
004E 1E            PUSH    DS              ;该中断向量指向一个11字节的磁盘参
                                           ;数表
004F 56            PUSH    SI              ;取到后压入堆栈中保存
0050 16            PUSH    SS
0051 53            PUSH    BX              ;保存地址0000:0078H
0052 BF3E7C        MOV     DI,7C3E         ;7C3E-7C00=003EH,即偏移003EH,以下
                                           ;类推
0055 B90B00        MOV     CX,000B         ;磁盘参数表共11字节
0058 FC            CLD
0059 F3            REPZ
005A A4            MOVSB                   ;将磁盘参数表复制到0000:7C3EH处
005B 06            PUSH    ES
005C 1F            POP     DS              ;(DS)=0000H
005D C645FE0F      MOV     BYTE PTR [DI-02],0F ;修改参数表中"磁头定位时间"
0061 8B0E187C      MOV     CX,[7C18]       ;从BPB中取"每磁道扇区数"
0065 884DF9        MOV     [DI-07],CL      ;修改参数表中"每磁道扇区数"
0068 894702        MOV     [BX+02],AX      ;(AX)=0000H,修改1EH号中断向量
                                           ;(段址)
006B C7073E7C      MOV     WORD PTR [BX],7C3E ;修改1EH号中断向量(偏移),这
                                              ;样1EH号
006F FB            STI           ;中断向量的内容为0000:7C3EH,指向新的磁盘参
                                 ;数表
0070 CD13          INT     13              ;用新的磁盘参数表来复位磁盘
0072 7279          JB      00ED            ;出错则转出错处理
----------------------------------------------------------------------------
---
                                           ; 下面一段程序计算扇区位置
0074 33C0          XOR     AX,AX
0076 3906137C      CMP     [7C13],AX       ;偏移0013H处是Dos分区的总扇区数
007A 7408          JZ      0084            ;为零表示大硬盘?
007C 8B0E137C      MOV     CX,[7C13]       ;不为0则取出来放到偏移0020H处
0080 890E207C      MOV     [7C20],CX       ;这个值本程序未用,似乎为IO.SYS准备
                                           ;的
0084 A0107C        MOV     AL,[7C10]       ;取FAT表的个数
0087 F726167C      MUL     WORD PTR [7C16] ;乘以一个FAT表所占的扇区数
008B 03061C7C      ADD     AX,[7C1C]       ;加上Dos分区前的扇区数(隐藏扇数,低
                                           ;位)
008F 13161E7C      ADC     DX,[7C1E]       ;                               高
                                           ;位)
0093 03060E7C      ADD     AX,[7C0E]       ;加上Dos分区内的保留扇区数(低位)
0097 83D200        ADC     DX,+00          ;                         (高位)
009A A3507C        MOV     [7C50],AX       ;根目录表的首扇的逻辑扇区号(低位)

009D 8916527C      MOV     [7C52],DX       ;                          (高位)

00A1 A3497C        MOV     [7C49],AX       ;此处放IO.SYS的首扇的逻辑扇区号(低
                                           ;位)
00A4 89164B7C      MOV     [7C4B],DX       ;                              (高
                                           ;位)
00A8 B82000        MOV     AX,0020         ;根目录表中每项占32字节
00AB F726117C      MUL     WORD PTR [7C11] ;乘以根目录表中的项数
00AF 8B1E0B7C      MOV     BX,[7C0B]       ;取"每扇区的字节数"
00B3 03C3          ADD     AX,BX           ;这两条指令是为了取整
00B5 48            DEC     AX
00B6 F7F3          DIV     BX              ;除以每扇字节数,得到根目录所占扇区
                                           ;数
00B8 0106497C      ADD     [7C49],AX       ;得到根目录表后首扇的逻辑扇区号(低
                                           ;位)
00BC 83164B7C00    ADC     WORD PTR [7C4B],+00  ;                         (高
                                                ;位)
----------------------------------------------------------------------------
---
                       ;下面一段程序在根目录表中找系统文件IO.SYS和MSDOS.SYS
00C1 BB0005        MOV     BX,0500         ;内存缓冲区的偏移值
00C4 8B16527C      MOV     DX,[7C52]       ;取根目录表的首扇的逻辑扇区号(高
                                           ;位)
00C8 A1507C        MOV     AX,[7C50]       ;                            (低
                                           ;位)
00CB E89200        CALL    0160            ;将逻辑扇区号转换为物理扇区号
00CE 721D          JB      00ED            ;出错则转出错处理
00D0 B001          MOV     AL,01
00D2 E8AC00        CALL    0181            ;读一个扇区到内存(根目录的首扇)
00D5 7216          JB      00ED            ;出错处理
00D7 8BFB          MOV     DI,BX           ;内存缓冲区的首址
00D9 B90B00        MOV     CX,000B         ;比较11个字节
00DC BEE67D        MOV     SI,7DE6         ;偏移01E6处是串"IO      SYS",长11
                                           ;字节
00DF F3            REPZ
00E0 A6            CMPSB                   ;看第一项是否为IO.SYS
00E1 750A          JNZ     00ED            ;不是则出错
00E3 8D7F20        LEA     DI,[BX+20]      ;跳过32字节就指向第二项
00E6 B90B00        MOV     CX,000B         ;比较11个字节
00E9 F3            REPZ
00EA A6            CMPSB                   ;看第二项是否为MSDOS.SYS
00EB 7418          JZ      0105            ;是则两个文件都已找到,跳过出错处理

----------------------------------------------------------------------------
--
                                           ;下面一段进行出错处理
00ED BE9E7D        MOV     SI,7D9E         ;偏移019EH处是串"Non system disk.
                                           ;.."
00F0 E85F00        CALL    0152            ;显示字符串
00F3 33C0          XOR     AX,AX
00F5 CD16          INT     16              ;等待任一键按下
00F7 5E            POP     SI
00F8 1F            POP     DS              ;得到1EH号中断向量的地址0000:0078H
00F9 8F04          POP     [SI]
00FB 8F4402        POP     [SI+02]         ;恢复1EH号中断向量的内容
00FE CD19          INT     19              ;自举
0100 58            POP     AX
0101 58            POP     AX
0102 58            POP     AX              ;清理堆栈
0103 EBE8          JMP     00ED            ;再次试图起动
----------------------------------------------------------------------------
--
                       ;下面读入IO.SYS的头3个扇区到内存0000:0700H处
0105 8B471A        MOV     AX,[BX+1A]      ;从根目录表第一项中取IO.SYS的首簇
                                           ;号
0108 48            DEC     AX
0109 48            DEC     AX              ;首簇号减二
010A 8A1E0D7C      MOV     BL,[7C0D]       ;取每簇的扇区数
010E 32FF          XOR     BH,BH
0110 F7E3          MUL     BX              ;(首簇号 - 2)乘以 每簇的扇区数
0112 0306497C      ADD     AX,[7C49]       ;相加后得到IO.SYS的首扇的逻辑扇区
                                           ;号
0116 13164B7C      ADC     DX,[7C4B]
011A BB0007        MOV     BX,0700         ;内存缓冲区的偏移值
011D B90300        MOV     CX,0003         ;循环计数初值,读3个扇区
0120 50            PUSH    AX              ;逻辑扇区号进栈(低位)
0121 52            PUSH    DX              ;              (高位)
0122 51            PUSH    CX              ;循环计数器进栈
0123 E83A00        CALL    0160            ;逻辑扇区号转换为物理扇区号
0126 72D8          JB      0100            ;出错处理
0128 B001          MOV     AL,01
012A E85400        CALL    0181            ;读一个扇区到内存缓冲区
012D 59            POP     CX              ;循环计数出栈
012E 5A            POP     DX
012F 58            POP     AX              ;逻辑扇区号出栈
0130 72BB          JB      00ED            ;读盘出错处理
0132 050100        ADD     AX,0001
0135 83D200        ADC     DX,+00          ;下一个扇区
0138 031E0B7C      ADD     BX,[7C0B]       ;缓冲区指针移动一个扇区的大小
013C E2E2          LOOP    0120            ;循环读入三个扇区
013E 8A2E157C      MOV     CH,[7C15]       ;取"磁盘介质描述",传给IO.SYS
0142 8A16247C      MOV     DL,[7C24]       ;取"系统文件所在的驱动器号"
0146 8B1E497C      MOV     BX,[7C49]       ;取IO.SYS的首扇的逻辑扇区号
014A A14B7C        MOV     AX,[7C4B]
014D EA00007000    JMP     0070:0000       ;执行IO.SYS,引导完毕
----------------------------------------------------------------------------
--
                                           ;显示字符串的子程序
0152 AC            LODSB                   ;从串中取一个字符
0153 0AC0          OR      AL,AL
0155 7429          JZ      0180            ;为0则已到串尾,返回(共用RET指令)
0157 B40E          MOV     AH,0E
0159 BB0700        MOV     BX,0007
015C CD10          INT     10              ;显示该字符
015E EBF2          JMP     0152            ;循环显示下一个
----------------------------------------------------------------------------

                                           ;将逻辑扇区号转换为物理扇区号的子
                                           ;程序
0160 3B16187C      CMP     DX,[7C18]       ;这两条指令是为了避免第二次除法时
                                           ;除数
0164 7319          JNB     017F            ;为0
0166 F736187C      DIV     WORD PTR [7C18] ;逻辑扇取号除以每道扇区数,商(AX)=
                                           ;总磁
016A FEC2          INC     DL              ;道数,余数(DX)再加一即为扇区号,因
                                           ;为扇
016C 88164F7C      MOV     [7C4F],DL       ;区号是从1开始的,而不是从0开始
0170 33D2          XOR     DX,DX
0172 F7361A7C      DIV     WORD PTR [7C1A] ;总磁道数(AX)再除以面数,所得的
0176 8816257C      MOV     [7C25],DL       ;余数(DX)=面号(即磁头号)
017A A34D7C        MOV     [7C4D],AX       ;商(AX)=磁道号
017D F8            CLC
017E C3            RET                     ;正常返回
017F F9            STC
0180 C3            RET                     ;异常返回
----------------------------------------------------------------------------
-
                                           ;读一个扇区的子程序
0181 B402          MOV     AH,02           ;读功能调用
0183 8B164D7C      MOV     DX,[7C4D]       ;需要的入口参数如下:
0187 B106          MOV     CL,06           ;(DL)=驱动器号
0189 D2E6          SHL     DH,CL           ;(DH)=面号
018B 0A364F7C      OR      DH,[7C4F]       ;(CH)=磁道号
018F 8BCA          MOV     CX,DX           ;(CL)=扇区号(第6,7位为磁道号的高2
                                           ;位)
0191 86E9          XCHG    CH,CL           ;(AL)=要读的扇区数
0193 8A16247C      MOV     DL,[7C24]       ;(ES:BX)=缓冲区首址
0197 8A36257C      MOV     DH,[7C25]
019B CD13          INT     13
019D C3            RET
----------------------------------------------------------------------------
---
0190                                            0D 0A                 ..
01A0  4E 6F 6E 2D 53 79 73 74-65 6D 20 64 69 73 6B 20   Non-System disk
01B0  6F 72 20 64 69 73 6B 20-65 72 72 6F 72 0D 0A 52   or disk error..R
01C0  65 70 6C 61 63 65 20 61-6E 64 20 70 72 65 73 73   eplace and press
01D0  20 61 6E 79 20 6B 65 79-20 77 68 65 6E 20 72 65    any key when re
01E0  61 64 79 0D 0A 00 49 4F-20 20 20 20 20 20 53 59   ady...IO      SY
01F0  53 4D 53 44 4F 53 20 20-20 53 59 53 00 00 55 AA   SMSDOS   SYS..U.

--

(本文采用S-Term文章拷贝脚本拷贝)
==================================================


   此帖被 +2 点积分    点击查看详情   
评分人:【 zzz19760225 分数: +2  时间:2017-10-31 16:36




ko20010214
=================================
大功告成,打个Kiss!
ko20010214@MSN.com
神州优雅Q300C
Intel CeleronM 370处理器 | 256MbDDR内存
40G硬盘 | USB2.0 | IEEE 1394
13.3 ' WXGA 宽屏(16:10) | COMBO光驱
10/100M网卡 | 四合一读卡器
2003-6-1 00:00
查看资料  发送邮件  发短消息 网志  OICQ (16959102)  编辑帖子  回复  引用回复

请注意:您目前尚未注册或登录,请您注册登录以使用论坛的各项功能,例如发表和回复帖子等。


可打印版本 | 推荐给朋友 | 订阅主题 | 收藏主题



论坛跳转: