第8章 硬盘和显卡的访问与控制

首先声明,这一章非常重要,如果刚开始读不懂,读不下去,一定要坚持,还有读这本书的一个要求是王爽《汇编语言》看两遍,并做完所有的课后实验。这一章其实是操作系统的的加载和引导过程。其中涉及的有硬盘读写,程序加载,程序重定位等,令人沮丧的是,这三个东西都不是那么容易,你有可能看了几遍还是不十分明晰,令人高兴的是,操作系统引导只需要这三个重要的东西,因为下一步就是进入操作系统的“森林”了。这些东西不难,你感觉难只是不熟悉而已。本章代码不完全明白熟悉,不要进入下一章,因为遗留的疑问会让你更弄不明白后边的问题。

下面分而治之,结合载入程序代码请思考如下问题:

a)程序在哪里?

程序放在100扇区

b)程序存移动到哪里?

0x10000 这个地址没有什么特别的意思,你可以放到其它合适的地址,唯一的要求是该地址的最低4 位必须是 0,换句话说,加载的起始地址必须是16 字节对齐的,这样将来才能形成一个 有效的段地址。

c)怎样读磁盘?

read_hard_disk0

d)怎样放到目的内存?

e)怎样“搬运”程序?

f)为什么重定位,怎样重定位?

载入程序

         ;代码清单8-1;文件名:c08_mbr.asm;文件说明:硬盘主引导扇区代码(加载程序) ;创建日期:2011-5-5 18:17app_lba_start equ 100           ;声明常数(用户程序起始逻辑扇区号);常数的声明不会占用汇编地址SECTION mbr align=16 vstart=0x7c00                                     ;设置堆栈段和栈指针 mov ax,0      mov ss,axmov sp,axmov ax,[cs:phy_base]            ;计算用于加载用户程序的逻辑段地址 mov dx,[cs:phy_base+0x02]mov bx,16        div bx            mov ds,ax                       ;令DS和ES指向该段以进行操作mov es,ax                        ;以下读取程序的起始部分 xor di,dimov si,app_lba_start            ;程序在硬盘上的起始逻辑扇区号 xor bx,bx                       ;加载到DS:0x0000处 call read_hard_disk_0;以下判断整个程序有多大mov dx,[2]                      ;曾经把dx写成了ds,花了二十分钟排错 mov ax,[0]mov bx,512                      ;512字节每扇区div bxcmp dx,0jnz @1                          ;未除尽,因此结果比实际扇区数少1 dec ax                          ;已经读了一个扇区,扇区总数减1 @1:cmp ax,0                        ;考虑实际长度小于等于512个字节的情况 jz direct;读取剩余的扇区push ds                         ;以下要用到并改变DS寄存器 mov cx,ax                       ;循环次数(剩余扇区数)@2:mov ax,dsadd ax,0x20                     ;得到下一个以512字节为边界的段地址mov ds,ax  xor bx,bx                       ;每次读时,偏移地址始终为0x0000 inc si                          ;下一个逻辑扇区 call read_hard_disk_0loop @2                         ;循环读,直到读完整个功能程序 pop ds                          ;恢复数据段基址到用户程序头部段 ;计算入口点代码段基址 direct:mov dx,[0x08]mov ax,[0x06]call calc_segment_basemov [0x06],ax                   ;回填修正后的入口点代码段基址 ;开始处理段重定位表mov cx,[0x0a]                   ;需要重定位的项目数量mov bx,0x0c                     ;重定位表首地址realloc:mov dx,[bx+0x02]                ;32位地址的高16位 mov ax,[bx]call calc_segment_basemov [bx],ax                     ;回填段的基址add bx,4                        ;下一个重定位项(每项占4个字节) loop realloc jmp far [0x04]                  ;转移到用户程序  ;-------------------------------------------------------------------------------
read_hard_disk_0:                        ;从硬盘读取一个逻辑扇区;输入:DI:SI=起始逻辑扇区号;      DS:BX=目标缓冲区地址push axpush bxpush cxpush dxmov dx,0x1f2mov al,1out dx,al                       ;读取的扇区数inc dx                          ;0x1f3mov ax,siout dx,al                       ;LBA地址7~0inc dx                          ;0x1f4mov al,ahout dx,al                       ;LBA地址15~8inc dx                          ;0x1f5mov ax,diout dx,al                       ;LBA地址23~16inc dx                          ;0x1f6mov al,0xe0                     ;LBA28模式,主盘or al,ah                        ;LBA地址27~24out dx,alinc dx                          ;0x1f7mov al,0x20                     ;读命令out dx,al.waits:in al,dxand al,0x88cmp al,0x08jnz .waits                      ;不忙,且硬盘已准备好数据传输 mov cx,256                      ;总共要读取的字数mov dx,0x1f0.readw:in ax,dxmov [bx],axadd bx,2loop .readwpop dxpop cxpop bxpop axret;-------------------------------------------------------------------------------
calc_segment_base:                       ;计算16位段地址;输入:DX:AX=32位物理地址;返回:AX=16位段基地址 push dx                          add ax,[cs:phy_base]adc dx,[cs:phy_base+0x02]shr ax,4ror dx,4and dx,0xf000or ax,dxpop dxret;-------------------------------------------------------------------------------phy_base dd 0x10000             ;用户程序被加载的物理起始地址times 510-($-$$) db 0db 0x55,0xaa




用户程序

         ;代码清单8-2;文件名:c08.asm;文件说明:用户程序 ;创建日期:2011-5-5 18:17;===============================================================================
SECTION header vstart=0                     ;定义用户程序头部段 program_length  dd program_end          ;程序总长度[0x00];用户程序入口点code_entry      dw start                ;偏移地址[0x04]dd section.code_1.start ;段地址[0x06] realloc_tbl_len dw (header_end-code_1_segment)/4;段重定位表项个数[0x0a];段重定位表           code_1_segment  dd section.code_1.start ;[0x0c]code_2_segment  dd section.code_2.start ;[0x10]data_1_segment  dd section.data_1.start ;[0x14]data_2_segment  dd section.data_2.start ;[0x18]stack_segment   dd section.stack.start  ;[0x1c]header_end:                ;===============================================================================
SECTION code_1 align=16 vstart=0         ;定义代码段1(16字节对齐) 
put_string:                              ;显示串(0结尾)。;输入:DS:BX=串地址mov cl,[bx]or cl,cl                        ;cl=0 ?jz .exit                        ;是的,返回主程序 call put_charinc bx                          ;下一个字符 jmp put_string.exit:ret;-------------------------------------------------------------------------------
put_char:                                ;显示一个字符;输入:cl=字符asciipush axpush bxpush cxpush dxpush dspush es;以下取当前光标位置mov dx,0x3d4mov al,0x0eout dx,almov dx,0x3d5in al,dx                        ;高8位 mov ah,almov dx,0x3d4mov al,0x0fout dx,almov dx,0x3d5in al,dx                        ;低8位 mov bx,ax                       ;BX=代表光标位置的16位数cmp cl,0x0d                     ;回车符?jnz .put_0a                     ;不是。看看是不是换行等字符 mov ax,bx                       ;此句略显多余,但去掉后还得改书,麻烦 mov bl,80                       div blmul blmov bx,axjmp .set_cursor.put_0a:cmp cl,0x0a                     ;换行符?jnz .put_other                  ;不是,那就正常显示字符 add bx,80jmp .roll_screen.put_other:                             ;正常显示字符mov ax,0xb800mov es,axshl bx,1mov [es:bx],cl;以下将光标位置推进一个字符shr bx,1add bx,1.roll_screen:cmp bx,2000                     ;光标超出屏幕?滚屏jl .set_cursormov ax,0xb800mov ds,axmov es,axcldmov si,0xa0mov di,0x00mov cx,1920rep movswmov bx,3840                     ;清除屏幕最底一行mov cx,80.cls:mov word[es:bx],0x0720add bx,2loop .clsmov bx,1920.set_cursor:mov dx,0x3d4mov al,0x0eout dx,almov dx,0x3d5mov al,bhout dx,almov dx,0x3d4mov al,0x0fout dx,almov dx,0x3d5mov al,blout dx,alpop espop dspop dxpop cxpop bxpop axret;-------------------------------------------------------------------------------start:;初始执行时,DS和ES指向用户程序头部段mov ax,[stack_segment]           ;设置到用户程序自己的堆栈 mov ss,axmov sp,stack_endmov ax,[data_1_segment]          ;设置到用户程序自己的数据段mov ds,axmov bx,msg0call put_string                  ;显示第一段信息 push word [es:code_2_segment]mov ax,beginpush ax                          ;可以直接push begin,80386+retf                             ;转移到代码段2执行 continue:mov ax,[es:data_2_segment]       ;段寄存器DS切换到数据段2 mov ds,axmov bx,msg1call put_string                  ;显示第二段信息 jmp $ ;===============================================================================
SECTION code_2 align=16 vstart=0          ;定义代码段2(16字节对齐)begin:push word [es:code_1_segment]mov ax,continuepush ax                          ;可以直接push continue,80386+retf                             ;转移到代码段1接着执行 ;===============================================================================
SECTION data_1 align=16 vstart=0msg0 db '  This is NASM - the famous Netwide Assembler. 'db 'Back at SourceForge and in intensive development! 'db 'Get the current versions from http://www.nasm.us/.'db 0x0d,0x0a,0x0d,0x0adb '  Example code for calculate 1+2+...+1000:',0x0d,0x0a,0x0d,0x0adb '     xor dx,dx',0x0d,0x0adb '     xor ax,ax',0x0d,0x0adb '     xor cx,cx',0x0d,0x0adb '  @@:',0x0d,0x0adb '     inc cx',0x0d,0x0adb '     add ax,cx',0x0d,0x0adb '     adc dx,0',0x0d,0x0adb '     inc cx',0x0d,0x0adb '     cmp cx,1000',0x0d,0x0adb '     jle @@',0x0d,0x0adb '     ... ...(Some other codes)',0x0d,0x0a,0x0d,0x0adb 0;===============================================================================
SECTION data_2 align=16 vstart=0msg1 db '  The above contents is written by LeeChung. 'db '2011-05-06'db 0;===============================================================================
SECTION stack align=16 vstart=0resb 256stack_end:  ;===============================================================================
SECTION trail align=16
program_end:

实验现象:


实验体会:

本章包含的信息量特别大,重点关注四个东西:a)怎样读磁盘;b)怎样加载程序;c)怎样重定位;d)从bios启动到陷入用户程序的死循环整个流程cs:ip是怎样变化的,相应的,像字符串的显示也是比较复杂的,但不紧急,可以暂时不了解这些细节,放在平时慢慢揣摩熟悉。













本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/421009.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

和不安全的Android说再见,Google为它添加新铠甲

伴随着最近被曝出的Heartbleed漏洞,安全问题再次成为网络上的热点话题。虽然Android的安全性并没有外界传言的那样脆弱,但Google还是在继续为其增加防护措施。根据Android官方博客的消息,Google最近对Android上的应用程序验证机制进行了一次升…

face-recognition库(人脸识别库)安装,使用

一,起因 1,在某hub上下载了所谓“最简单的人脸识别项目”。 2,开始了face-recognition的安装之路。 3,人脸识别在win10上的使用,识别图像上面的人脸。 二,安装face-recognition库 1,我用的都…

python入门--字符串,切片

1,python关键字 2,代码的缩进(有冒号, 下一行空格) 注意下面 3,python多行语句 4,python空行 类与函数间空行,方法与方法间空行,函数与函数空行。 5,控制台输入 下…

第9章 中断和动态时钟显示

本章的第一个代码功能是在屏幕中间实时显示时间,其实现的思想是:cpu停机---》时间每秒更新一次,每一次更新都会引起0x70中断(0x70中断的内容已经被我们更改为显示当前时间)---》中断唤醒cpu开始执行我们的中断程序知道…

linux 架构

最内层是我们的硬件,最外层是我们常用的各种应用,比如说使用firefox浏览器,打开evolution查看邮件,运行一个计算流体模型等等。硬件是我们的物质基础,而应用是我们所要奋斗的目标,但在两者之间,…

py学习,列表,判断,循环

列表 1.1 列表是什么 python 中可以通过组合一些值得到多种复合数据类型。其中最常用的是列表,可以通过方括号括起,逗号分隔的一组值(元素)得到,一个列表可以包含不同类型的元素,一个列表的元素各个类型相同…

现代软件工程讲义 3 代码规范与代码复审

请参考原址:http://www.cnblogs.com/xinz/archive/2011/11/20/2255971.html 第10章 代码规范与代码复审 在第9章中,同学们完成了WC程序,经过评比,九条的程序获得了第一名。这时,阿超说,现代软件产业经过几十…

HDU 2159 FATE (DP 二维费用背包)

题目链接 题意 : 中文题不详述。 思路 : 二维背包,dp[i][h]表示当前忍耐值为i的情况下,杀了h个怪得到的最大经验值,状态转移方程: dp[i][h] max(dp[i][h],dp[i-a[j].toler][h-1]a[j].exper) ; 1 //21592 #…

TinyFrame升级之五:全局缓存的设计及实现

在任何框架中,缓存都是不可或缺的一部分,本框架亦然。在这个框架中,我们的缓存分为两部分:内存缓存和单次请求缓存。简单说来,就是一个使用微软提供的MemoryCache做扩展,并提供全局唯一实例;另一…

问题思维导向

杨淼信息与计算机科学 跆拳道国家一级教练员 曾经tricking爱好者3 人赞同了该回答一个不算建议的建议 直接看,看到不懂的就查相应的知识(据说牛人都靠wiki活着)。 知识当有需求的时候,才会学的更快。 觉得这样做的好处是&#xff…

【Mail】telnet收发邮件过程

文章原址:http://www.cnblogs.com/h--d/p/6125993.html 作者写的真棒,按照步骤一步一步都可执行,我是在centos7服务器下进行的实验,命令完全和下面一样,注意的地方是: 1,centos7要安装telnet&a…

入门训练 Fibonacci数列

http://lx.lanqiao.org/problem.page?gpidT4入门训练 Fibonacci数列 时间限制:1.0s 内存限制:256.0MB问题描述Fibonacci数列的递推公式为:FnFn-1Fn-2,其中F1F21。 当n比较大时,Fn也非常大,现在我们想知…

python模块与包的导入

一&#xff0c;模块 1&#xff0c;下面是hello.py脚本(模块)&#xff0c;有两求Fibonacci的函数. def fib1():a, b 0, 1print(0-10内:)while a < 10:print(a, end )a, b b, a bprint()def fib2(n):a, b 0, 1L []while a < n:L.append(a)a, b b, a bprint(L)retu…

第11章 进入保护模式

对照代码重点理解一下三个要点&#xff1a; 1,从实模式到保护模式的整体过程&#xff1b; 2&#xff0c;实模式怎样安装GDT&#xff1b; 3&#xff0c;从实模式怎样过渡到安全模式&#xff1b; 4&#xff0c;安全模式怎样寻址&#xff1b; 代码如下&#xff1a; ;代码清单1…

ASP.NET学习笔记之操作过滤器

转载于:https://www.cnblogs.com/kevin1988/p/3684442.html

第12章 存储器的保护

学习32位保护模式明显感觉比16位实模式复杂一些&#xff0c;其实两者之间所做的事情是一样的&#xff0c;无非是对内存的各种操作&#xff0c;对栈的各种操作&#xff0c;但是手段不一样&#xff0c;实模式是野蛮的&#xff0c;可以在任意地方读写&#xff08;比如代码区&#…

with open的使用

1, open(‘file’, mode) 第一个参数是包含文件名的字符串。第二个参数是另一个字符串&#xff0c;其中包含一些描述文件使用方式的字符。mode 可以是 ‘r’&#xff0c;表示文件只能读取&#xff0c;‘w’ 表示只能写入&#xff08;已存在的同名文件会被删除&#xff09;&…