中国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)  编辑帖子  回复  引用回复

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


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



论坛跳转: