汇编语言 第3版 王爽 检测点答案及详细解析
<div class="article-header-box"><div class="article-header">
<div class="article-title-box">
<h1 id="articleContentId" class="title-article">汇编语言 第3版 王爽 检测点答案及详细解析</h1>
</div>
<div class="article-info-box">
<div class="article-bar-top">转自:https://blog.csdn.net/qq_42777804/article/details/90512159</div>
</div>
</div>
</div>
<div id="article_content" class="article_content clearfix">
<div id="content_views" class="htmledit_views">
<h2>第一章 基础知识</h2>
<p>检测点1.1 </p>
<p>(1)1个CPU的寻址能力为8KB,那么它的地址总线的宽度为<span style="text-decoration: underline"> 13</span>位。</p>
<p>(2)1KB的存储器有<span style="text-decoration: underline"> 1024 </span>个存储单元,存储单元的编号从<span style="text-decoration: underline"> 0 </span>到<span style="text-decoration: underline"> 1023 </span>。</p>
<p>(3)1KB的存储器可以存储<span style="text-decoration: underline"> 8192(2^13) </span>个bit,<span style="text-decoration: underline"> 1024</span>个Byte。</p>
<p>(4)1GB是<span style="text-decoration: underline"> 1073741824 (2^30) </span>个Byte、1MB是<span style="text-decoration: underline"> 1048576(2^20) </span>个Byte、1KB是<span style="text-decoration: underline"> 1024(2^10)</span>个Byte。</p>
<p>(5)8080、8088、80296、80386的地址总线宽度分别为16根、20根、24根、32根,则它们的寻址能力分别为:<span style="text-decoration: underline"> 64 </span>(KB)、<span style="text-decoration: underline"> 1 </span>(MB)、<span style="text-decoration: underline"> 16 </span>(MB)、<span style="text-decoration: underline"> 4 </span>(GB)。</p>
<p>(6)8080、8088、8086、80286、80386的数据总线宽度分别为8根、8根、16根、16根、32根。则它们一次可以传送的数据为:<span style="text-decoration: underline"> 1 </span>(B)、<span style="text-decoration: underline"> 1 </span>(B)、<span style="text-decoration: underline"> 2 </span>(B)、<span style="text-decoration: underline"> 2 </span>(B)、<span style="text-decoration: underline"> 4 </span>(B)。</p>
<p>(7)从内存中读取1024字节的数据,8086至少要读<span style="text-decoration: underline"> 512 </span>次,80386至少要读<span style="text-decoration: underline"> 256 </span>次。</p>
<p>(8)在存储器中,数据和程序以<span style="text-decoration: underline"> 二进制 </span>形式存放。</p>
<p> </p>
<p>解题过程:</p>
<p>(1)1KB=1024B,8KB=1024B*8=2^N,N=13。</p>
<p>(2)存储器的容量是以字节为最小单位来计算的,1KB=1024B。</p>
<p>(3)8Bit=1Byte,1024Byte=1KB(1KB=1024B=1024B*8Bit)。</p>
<p>(4)1GB=1073741824B(即2^30)1MB=1048576B(即2^20)1KB=1024B(即2^10)。</p>
<p>(5)一个CPU有N根地址线,则可以说这个CPU的地址总线的宽度为N。这样的CPU最多可以寻找2的N次方个内存单元。(一个内存单元=1Byte)。</p>
<p>(6)8根数据总线一次可以传送8位二进制数据(即一个字节)。</p>
<p>(7)8086的数据总线宽度为16根(即一次传送的数据为2B)1024B/2B=512,同理1024B/4B=256。</p>
<p>(8)在存储器中指令和数据没有任何区别,都是二进制信息。</p>
<p> </p>
<p> </p>
<h2>第二章 寄存器</h2>
<p>检测点 2.1</p>
<p>(1) 写出每条汇编指令执行后相关寄存器中的值。</p>
<p>mov ax,62627 AX=F4A3H </p>
<p>mov ah,31H AX=31A3H </p>
<p>mov al,23H AX=3123H </p>
<p>add ax,ax AX=6246H </p>
<p>mov bx,826CH BX=826CH </p>
<p>mov cx,ax CX=6246H </p>
<p>mov ax,bx AX=826CH </p>
<p>add ax,bx AX=04D8H </p>
<p>mov al,bh AX=0482H </p>
<p>mov ah,bl AX=6C82H </p>
<p>add ah,ah AX=D882H </p>
<p>add al,6 AX=D888H </p>
<p>add al,al AX=D810H </p>
<p>mov ax,cx AX=6246H</p>
<p> </p>
<p>Microsoft(R) Windows DOS</p>
<p>(C)Copyright Microsoft Corp 1990-2001.</p>
<p>C:\DOCUME~1\ADMINI~1>debug</p>
<p>-a</p>
<p>0C1C:0100 mov ax,f4a3</p>
<p>0C1C:0103 mov ah,31</p>
<p>0C1C:0105 mov al,23</p>
<p>0C1C:0107 add ax,ax</p>
<p>0C1C:0109 mov bx,826c</p>
<p>0C1C:010C mov cx,ax</p>
<p>0C1C:010E mov ax,bx</p>
<p>0C1C:0110 add ax,bx</p>
<p>0C1C:0112 mov al,bh</p>
<p>0C1C:0114 mov ah,bl</p>
<p>0C1C:0116 add ah,ah</p>
<p>0C1C:0118 add al,6</p>
<p>0C1C:011A add al,al</p>
<p>0C1C:011C mov ax,cx</p>
<p>0C1C:011E</p>
<p>-r</p>
<p>AX=0000 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=0100 NV UP EI PL NZ NA PO NC</p>
<p>0C1C:0100 B8A3F4 MOV AX,F4A3</p>
<p>-t</p>
<p>AX=F4A3 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=0103 NV UP EI PL NZ NA PO NC</p>
<p>0C1C:0103 B431 MOV AH,31</p>
<p>-t</p>
<p>AX=31A3 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=0105 NV UP EI PL NZ NA PO NC</p>
<p>0C1C:0105 B023 MOV AL,23</p>
<p>-t</p>
<p>AX=3123 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=0107 NV UP EI PL NZ NA PO NC</p>
<p>0C1C:0107 01C0 ADD AX,AX</p>
<p>-t</p>
<p>AX=6246 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=0109 NV UP EI PL NZ NA PO NC</p>
<p>0C1C:0109 BB6C82 MOV BX,826C</p>
<p>-t</p>
<p>AX=6246 BX=826C CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=010C NV UP EI PL NZ NA PO NC</p>
<p>0C1C:010C 89C1 MOV CX,AX</p>
<p>-t</p>
<p>AX=6246 BX=826C CX=6246 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=010E NV UP EI PL NZ NA PO NC</p>
<p>0C1C:010E 89D8 MOV AX,BX</p>
<p>-t</p>
<p>AX=826C BX=826C CX=6246 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=0110 NV UP EI PL NZ NA PO NC</p>
<p>0C1C:0110 01D8 ADD AX,BX</p>
<p>-t</p>
<p>AX=04D8 BX=826C CX=6246 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=0112 OV UP EI PL NZ AC PE CY</p>
<p>0C1C:0112 88F8 MOV AL,BH</p>
<p>-t</p>
<p>AX=0482 BX=826C CX=6246 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=0114 OV UP EI PL NZ AC PE CY</p>
<p>0C1C:0114 88DC MOV AH,BL</p>
<p>-t</p>
<p>AX=6C82 BX=826C CX=6246 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=0116 OV UP EI PL NZ AC PE CY</p>
<p>0C1C:0116 00E4 ADD AH,AH</p>
<p>-t</p>
<p>AX=D882 BX=826C CX=6246 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=0118 OV UP EI NG NZ AC PE NC</p>
<p>0C1C:0118 0406 ADD AL,06</p>
<p>-t</p>
<p>AX=D888 BX=826C CX=6246 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=011A NV UP EI NG NZ NA PE NC</p>
<p>0C1C:011A 00C0 ADD AL,AL</p>
<p>-t</p>
<p>AX=D810 BX=826C CX=6246 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=011C OV UP EI PL NZ AC PO CY</p>
<p>0C1C:011C 89C8 MOV AX,CX</p>
<p>-t</p>
<p>AX=6246 BX=826C CX=6246 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=011E OV UP EI PL NZ AC PO CY</p>
<p>0C1C:011E 0B0C OR CX, DS:0000=20CD</p>
<p>-q</p>
<p> </p>
<p>检测点2.1</p>
<p>(2) 只能使用目前学过的汇编指令,最多使用4条指令,编程计算2的4次方。 </p>
<p>mov ax,2 AX=2 </p>
<p>add ax,ax AX=4 </p>
<p>add ax,ax AX=8 </p>
<p>add ax,ax AX=16 </p>
<p> </p>
<p> </p>
<p>Microsoft(R) Windows DOS</p>
<p>(C)Copyright Microsoft Corp 1990-2001.</p>
<p>C:\DOCUME~1\ADMINI~1>debug</p>
<p>-a</p>
<p>0C1C:0100 mov ax,2</p>
<p>0C1C:0103 add ax,ax</p>
<p>0C1C:0105 add ax,ax</p>
<p>0C1C:0107 add ax,ax</p>
<p>0C1C:0109</p>
<p>-r</p>
<p>AX=0000 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=0100 NV UP EI PL NZ NA PO NC</p>
<p>0C1C:0100 B80200 MOV AX,0002</p>
<p>-t</p>
<p>AX=0002 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=0103 NV UP EI PL NZ NA PO NC</p>
<p>0C1C:0103 01C0 ADD AX,AX</p>
<p>-t</p>
<p>AX=0004 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=0105 NV UP EI PL NZ NA PO NC</p>
<p>0C1C:0105 01C0 ADD AX,AX</p>
<p>-t</p>
<p>AX=0008 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=0107 NV UP EI PL NZ NA PO NC</p>
<p>0C1C:0107 01C0 ADD AX,AX</p>
<p>-t</p>
<p>AX=0010 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=0109 NV UP EI PL NZ AC PO NC</p>
<p>0C1C:0109 20881615 AND ,CL DS:1516=00</p>
<p>-q</p>
<p> </p>
<p>检测点2.2</p>
<p>(1) 给定段地址为0001H,仅通过变化偏移地址寻址,CPU的寻址范围为<span style="text-decoration: underline"> 0010H </span>到<span style="text-decoration: underline"> 1000FH </span>。</p>
<p> </p>
<p>解题过程:</p>
<p>物理地址=SA*16+EA </p>
<p>EA的变化范围为0h~ffffh </p>
<p>物理地址范围为(SA*16+0h)~(SA*16+ffffh) </p>
<p>现在SA=0001h,那么寻址范围为 </p>
<p>(0001h*16+0h)~(0001h*16+ffffh) </p>
<p>=0010h~1000fh </p>
<p> </p>
<p>检测点2.2</p>
<p>(2) 有一数据存放在内存20000H单元中,现给定段地址为SA,若想用偏移地址寻到此单元。则SA应满足的条件是:最小为<span style="text-decoration: underline"> 1001H </span>,最大为<span style="text-decoration: underline"> 2000H </span>。</p>
<p>当段地址给定为 1001H 以下和 2000H 以上,CPU无论怎么变化偏移地址都无法寻到20000H单元。</p>
<p> </p>
<p> </p>
<p>解题过程:</p>
<p>物理地址=SA*16+EA </p>
<p>20000h=SA*16+EA </p>
<p>SA=(20000h-EA)/16=2000h-EA/16 </p>
<p>EA取最大值时,SA=2000h-ffffh/16=1001h,SA为最小值 </p>
<p>EA取最小值时,SA=2000h-0h/16=2000h,SA为最大值 </p>
<p> </p>
<p>这里的ffffH/16=fffh是通过WIN自带计算器算的</p>
<p>按位移来算确实应该为fff.fh,这里小数点后的f应该是省略了</p>
<p>单就除法来说,应有商和余数,但此题要求的是地址最大和最小,所以余数忽略了</p>
<p> </p>
<p>如果根据位移的算法(段地址*16=16进制左移一位),小数点后应该是不能省略的</p>
<p>我们可以反过来再思考下,如果SA为1000h的话,小数点后省略</p>
<p>SA=1000h,EA取最大ffffh,物理地址为1ffffh,将无法寻到20000H单元</p>
<p>这道题不应看成是单纯的计算题</p>
<p> </p>
<p>检测点2.3</p>
<p>下面的3条指令执行后,cpu几次修改IP?都是在什么时候?最后IP中的值是多少? </p>
<p>mov ax,bx </p>
<p>sub ax,ax </p>
<p>jmp ax </p>
<p> </p>
<p>答:一共修改四次</p>
<p>第一次:读取mov ax,bx之后 </p>
<p>第二次:读取sub ax,ax之后 </p>
<p>第三次:读取jmp ax之后 </p>
<p>第四次:执行jmp ax修改IP </p>
<p>最后IP的值为0000H,因为最后ax中的值为0000H,所以IP中的值也为0000H </p>
<p> </p>
<p>检测点2.3</p>
<p>下面的3条指令执行后,cpu几次修改IP?都是在什么时候?最后IP中的值是多少? </p>
<p>mov ax,bx </p>
<p>sub ax,ax </p>
<p>jmp ax </p>
<p> </p>
<p>答:一共修改四次</p>
<p>第一次:读取mov ax,bx之后 </p>
<p>第二次:读取sub ax,ax之后 </p>
<p>第三次:读取jmp ax之后 </p>
<p>第四次:执行jmp ax修改IP </p>
<p>最后IP的值为0000H,因为最后ax中的值为0000H,所以IP中的值也为0000H </p>
<p> </p>
<h3>第三章 寄存器(内存访问)</h3>
<p>检测点3.1</p>
<p>(1) 在DEBUG中,用 "D 0:0 lf" 查看内存,结果如下: </p>
<p>0000:0000 70 80 F0 30 EF 60 30 E2-00 80 80 12 66 20 22 60 </p>
<p>0000:0010 62 26 E6 D6 CC 2E 3C 3B-AB BA 00 00 26 06 66 88 </p>
<p>下面的程序执行前,AX=0,BX=0,写出每条汇编指令执行完后相关寄存器中的值</p>
<p>mov ax,1</p>
<p>mov ds,ax</p>
<p>mov ax, ax=<span style="text-decoration: underline"> 2662H </span></p>
<p>mov bx, bx=<span style="text-decoration: underline"> E626H </span></p>
<p>mov ax,bx ax=<span style="text-decoration: underline"> E626H </span></p>
<p>mov ax, ax=<span style="text-decoration: underline"> 2662H </span></p>
<p>mov bx, bx=<span style="text-decoration: underline"> D6E6H </span></p>
<p>add ax,bx ax=<span style="text-decoration: underline"> FD48H </span></p>
<p>add ax, ax=<span style="text-decoration: underline"> 2C14H </span></p>
<p>mov ax,0 ax=<span style="text-decoration: underline"> 0 </span></p>
<p>mov al, ax=<span style="text-decoration: underline"> 00e6H </span></p>
<p>mov bx,0 bx=<span style="text-decoration: underline"> 0 </span></p>
<p>mov bl, bx=<span style="text-decoration: underline"> 0026H </span></p>
<p>add al,bl ax=<span style="text-decoration: underline"> 000CH </span></p>
<p> </p>
<p>用DEBUG进行验证:</p>
<p>Microsoft(R) Windows DOS</p>
<p>(C)Copyright Microsoft Corp 1990-2001.</p>
<p>C:\DOCUME~1\000>debug</p>
<p>-e 0000:0</p>
<p>0000:0000 68.70 10.80 A7.f0 00.30 8B.ef 01.60 70.30 00.e2</p>
<p>0000:0008 16.00 00.80 AF.80 03.12 8B.66 01.20 70.22 00.60</p>
<p>0000:0010 8B.62 01.26 70.e6 00.d6 B9.cc 06.2e 14.3c 02.3b</p>
<p>0000:0018 40.ab 07.ba 14.00 02.00 FF.26 03.06 14.66 02.88</p>
<p>-d 0000:0 1f</p>
<p>0000:0000 70 80 F0 30 EF 60 30 E2-00 80 80 12 66 20 22 60 p..0.`0.....f "`</p>
<p>0000:0010 62 26 E6 D6 CC 2E 3C 3B-AB BA 00 00 26 06 66 88 b&....<;....&.f.</p>
<p>-a</p>
<p>0DB4:0100 mov ax,1</p>
<p>0DB4:0103 mov ds,ax</p>
<p>0DB4:0105 mov ax,</p>
<p>0DB4:0108 mov bx,</p>
<p>0DB4:010C mov ax,bx</p>
<p>0DB4:010E mov ax,</p>
<p>0DB4:0111 mov bx,</p>
<p>0DB4:0115 add ax,bx</p>
<p>0DB4:0117 add ax,</p>
<p>0DB4:011B mov ax,0</p>
<p>0DB4:011E mov al,</p>
<p>0DB4:0121 mov bx,0</p>
<p>0DB4:0124 mov bl,</p>
<p>0DB4:0128 add al,bl</p>
<p>0DB4:012A</p>
<p>-r</p>
<p>AX=0000 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0DB4 ES=0DB4 SS=0DB4 CS=0DB4 IP=0100 NV UP EI PL NZ NA PO NC</p>
<p>0DB4:0100 B80100 MOV AX,0001</p>
<p>-t</p>
<p>AX=0001 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0DB4 ES=0DB4 SS=0DB4 CS=0DB4 IP=0103 NV UP EI PL NZ NA PO NC</p>
<p>0DB4:0103 8ED8 MOV DS,AX</p>
<p>-t</p>
<p>AX=0001 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0001 ES=0DB4 SS=0DB4 CS=0DB4 IP=0105 NV UP EI PL NZ NA PO NC</p>
<p>0DB4:0105 A10000 MOV AX, DS:0000=2662</p>
<p>-t</p>
<p>AX=2662 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0001 ES=0DB4 SS=0DB4 CS=0DB4 IP=0108 NV UP EI PL NZ NA PO NC</p>
<p>0DB4:0108 8B1E0100 MOV BX, DS:0001=E626</p>
<p>-t</p>
<p>AX=2662 BX=E626 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0001 ES=0DB4 SS=0DB4 CS=0DB4 IP=010C NV UP EI PL NZ NA PO NC</p>
<p>0DB4:010C 89D8 MOV AX,BX</p>
<p>-t</p>
<p>AX=E626 BX=E626 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0001 ES=0DB4 SS=0DB4 CS=0DB4 IP=010E NV UP EI PL NZ NA PO NC</p>
<p>0DB4:010E A10000 MOV AX, DS:0000=2662</p>
<p>-t</p>
<p>AX=2662 BX=E626 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0001 ES=0DB4 SS=0DB4 CS=0DB4 IP=0111 NV UP EI PL NZ NA PO NC</p>
<p>0DB4:0111 8B1E0200 MOV BX, DS:0002=D6E6</p>
<p>-t</p>
<p>AX=2662 BX=D6E6 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0001 ES=0DB4 SS=0DB4 CS=0DB4 IP=0115 NV UP EI PL NZ NA PO NC</p>
<p>0DB4:0115 01D8 ADD AX,BX</p>
<p>-t</p>
<p>AX=FD48 BX=D6E6 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0001 ES=0DB4 SS=0DB4 CS=0DB4 IP=0117 NV UP EI NG NZ NA PE NC</p>
<p>0DB4:0117 03060400 ADD AX, DS:0004=2ECC</p>
<p>-t</p>
<p>AX=2C14 BX=D6E6 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0001 ES=0DB4 SS=0DB4 CS=0DB4 IP=011B NV UP EI PL NZ AC PE CY</p>
<p>0DB4:011B B80000 MOV AX,0000</p>
<p>-t</p>
<p>AX=0000 BX=D6E6 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0001 ES=0DB4 SS=0DB4 CS=0DB4 IP=011E NV UP EI PL NZ AC PE CY</p>
<p>0DB4:011E A00200 MOV AL, DS:0002=E6</p>
<p>-t</p>
<p>AX=00E6 BX=D6E6 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0001 ES=0DB4 SS=0DB4 CS=0DB4 IP=0121 NV UP EI PL NZ AC PE CY</p>
<p>0DB4:0121 BB0000 MOV BX,0000</p>
<p>-t</p>
<p>AX=00E6 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0001 ES=0DB4 SS=0DB4 CS=0DB4 IP=0124 NV UP EI PL NZ AC PE CY</p>
<p>0DB4:0124 8A1E0C00 MOV BL, DS:000C=26</p>
<p>-t</p>
<p>AX=00E6 BX=0026 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0001 ES=0DB4 SS=0DB4 CS=0DB4 IP=0128 NV UP EI PL NZ AC PE CY</p>
<p>0DB4:0128 00D8 ADD AL,BL</p>
<p>-t</p>
<p>AX=000C BX=0026 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0001 ES=0DB4 SS=0DB4 CS=0DB4 IP=012A NV UP EI PL NZ NA PE CY</p>
<p>0DB4:012A C6061799FF MOV BYTE PTR ,FF DS:9917=9A</p>
<p>-q</p>
<p> </p>
<p>检测点3.1 </p>
<p>(2) 内存中的情况如图3.6所示</p>
<p>各寄存器的初始值:cs=2000h,ip=0,ds=1000h,ax=0,bx=0;</p>
<p>① 写出CPU执行的指令序列(用汇编指令写出)。</p>
<p>② 写出CPU执行每条指令后,CS、IP和相关寄存器的数值。</p>
<p>③ 再次体会:数据和程序有区别吗?如何确定内存中的信息哪些是数据,哪些是程序?</p>
<p><img src="https://blog.csdn.net/qq_42777804/article/details/%E6%A3%80%E6%B5%8B%E7%82%B93.1%282%29.files/image001.jpg" alt="" id="_x0000_i1025" class="has"></p>
<p><img src="https://img-blog.csdnimg.cn/20190527230028480.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzQyNzc3ODA0,size_16,color_FFFFFF,t_70" alt="" width="501" height="294" class="has"></p>
<p>图3.6内存情况示意</p>
<p> </p>
<div class="table-box">
<table border="1" cellspacing="0">
<tbody>
<tr>
<td colspan="2">
<p>指令序列</p>
</td>
<td>
<p>CS</p>
</td>
<td>
<p>IP</p>
</td>
<td>
<p>DS</p>
</td>
<td>
<p>AX</p>
</td>
<td>
<p>BX</p>
</td>
</tr>
<tr>
<td colspan="2">
<p>初始值</p>
</td>
<td>
<p>2000h</p>
</td>
<td>
<p>0</p>
</td>
<td>
<p>0</p>
</td>
<td>
<p>0</p>
</td>
<td>
<p>0</p>
</td>
</tr>
<tr>
<td>
<p>1</p>
</td>
<td>
<p>mov ax,6622h</p>
</td>
<td>
<p>2000h</p>
</td>
<td>
<p>3h</p>
</td>
<td>
<p>0</p>
</td>
<td>
<p>6622h</p>
</td>
<td>
<p>0</p>
</td>
</tr>
<tr>
<td>
<p>2</p>
</td>
<td>
<p>jmp 0ff0:0100</p>
</td>
<td>
<p>ff0h</p>
</td>
<td>
<p>100h</p>
</td>
<td>
<p>0</p>
</td>
<td>
<p>6622h</p>
</td>
<td>
<p>0</p>
</td>
</tr>
<tr>
<td>
<p>3</p>
</td>
<td>
<p>mov ax,2000h</p>
</td>
<td>
<p>ff0h</p>
</td>
<td>
<p>103h</p>
</td>
<td>
<p>0</p>
</td>
<td>
<p>2000h</p>
</td>
<td>
<p>0</p>
</td>
</tr>
<tr>
<td>
<p>4</p>
</td>
<td>
<p>mov ds,ax</p>
</td>
<td>
<p>ff0h</p>
</td>
<td>
<p>105h</p>
</td>
<td>
<p>2000h</p>
</td>
<td>
<p>2000h</p>
</td>
<td>
<p>0</p>
</td>
</tr>
<tr>
<td>
<p>5</p>
</td>
<td>
<p>mov ax,</p>
</td>
<td>
<p>ff0h</p>
</td>
<td>
<p>108h</p>
</td>
<td>
<p>2000h</p>
</td>
<td>
<p>c389h</p>
</td>
<td>
<p>0</p>
</td>
</tr>
<tr>
<td>
<p>6</p>
</td>
<td>
<p>mov ax,</p>
</td>
<td>
<p>ff0h</p>
</td>
<td>
<p>10bh</p>
</td>
<td>
<p>2000h</p>
</td>
<td>
<p>ea66h</p>
</td>
<td>
<p>0</p>
</td>
</tr>
</tbody>
</table>
</div>
<p> </p>
<p>检测点3.2</p>
<p>(1)补全下面的程序,使其可以将10000H-1000FH中的8个字,逆序拷贝到20000H-2000FH中。</p>
<p>mov ax,1000H </p>
<p>mov ds,ax </p>
<p><span style="text-decoration: underline">mov ax,2000H </span></p>
<p><span style="text-decoration: underline">mov ss,ax </span></p>
<p><span style="text-decoration: underline">mov sp,10h </span></p>
<p>push </p>
<p>push </p>
<p>push </p>
<p>push </p>
<p>push </p>
<p>push </p>
<p>push </p>
<p>push </p>
<p> </p>
<p>检测点3.2 </p>
<p>(2)补全下面的程序,使其可以将10000H-1000FH中的8个字,逆序拷贝到20000H-2000FH中。 </p>
<p>mov ax,2000H </p>
<p>mov ds,ax </p>
<p><span style="text-decoration: underline">mov ax,1000H</span></p>
<p><span style="text-decoration: underline">mov ss,ax </span></p>
<p><span style="text-decoration: underline">mov sp,0 </span></p>
<p>pop </p>
<p>pop </p>
<p>pop </p>
<p>pop </p>
<p>pop </p>
<p>pop </p>
<p>pop </p>
<p>pop </p>
<p> </p>
<h3>第六章 包含多个段的程序</h3>
<p>检测点6.1</p>
<p>(1)下面的程序实现依次用内存0:0~0:15单元中的内容改写程序中的数据,完成程序:</p>
<p>assume cs:codesg</p>
<p>codesg segment</p>
<p> dw 0123h,0456h,0789h,0abch,0defh,0fedh,0cbah,0987h</p>
<p>start: mov ax,0</p>
<p> mov ds,ax</p>
<p> mov bx,0</p>
<p> mov cx,8</p>
<p> s: mov ax,</p>
<p> <span style="text-decoration: underline"> mov cs:,ax</span></p>
<p> add bx,2</p>
<p> loop s</p>
<p> mov ax,4c00h</p>
<p> int 21h</p>
<p>codesg ends</p>
<p>end start</p>
<p> </p>
<p>C:\DOCUME~1\ADMINI~1>debug jc6-1.exe</p>
<p>-u</p>
<p>0C79:0010 B80000 MOV AX,0000</p>
<p>0C79:0013 8ED8 MOV DS,AX</p>
<p>0C79:0015 BB0000 MOV BX,0000</p>
<p>0C79:0018 B90800 MOV CX,0008</p>
<p>0C79:001B 8B07 MOV AX,</p>
<p>0C79:001D 2E CS:</p>
<p>0C79:001E 8907 MOV ,AX</p>
<p>0C79:0020 83C302 ADD BX,+02</p>
<p>0C79:0023 E2F6 LOOP 001B</p>
<p>0C79:0025 B8004C MOV AX,4C00</p>
<p>0C79:0028 CD21 INT 21</p>
<p>0C79:002A 7503 JNZ 002F</p>
<p>0C79:002C E97BFF JMP FFAA</p>
<p>0C79:002F 5E POP SI</p>
<p>-g 0025</p>
<p>AX=0680 BX=0010 CX=0000 DX=0000 SP=0000 BP=0000 SI=0000 DI=0000</p>
<p>DS=0000 ES=0C69 SS=0C79 CS=0C79 IP=0025 NV UP EI PL NZ AC PO NC</p>
<p>0C79:0025 B8004C MOV AX,4C00</p>
<p>-d 0:0 f</p>
<p>0000:0000 68 10 A7 00 BB 13 80 06-16 00 A5 03 B1 13 80 06 h...............</p>
<p>-d 0c79:0 f</p>
<p>0C79:0000 68 10 A7 00 BB 13 80 06-16 00 A5 03 B1 13 80 06 h...............</p>
<p>-t</p>
<p>AX=4C00 BX=0010 CX=0000 DX=0000 SP=0000 BP=0000 SI=0000 DI=0000</p>
<p>DS=0000 ES=0C69 SS=0C79 CS=0C79 IP=0028 NV UP EI PL NZ AC PO NC</p>
<p>0C79:0028 CD21 INT 21</p>
<p>-p</p>
<p>Program terminated normally</p>
<p>-q</p>
<p>C:\DOCUME~1\ADMINI~1></p>
<p> </p>
<p>检测点6.1</p>
<p>(2)下面的程序实现依次用内存0:0~0:15单元中的内容改写程序中的数据,数据的传送用栈来进行。栈空间设置在程序内。完成程序:</p>
<p>assume cs:codesg</p>
<p>codesg segment</p>
<p> dw 0123h,0456h,0789h,0abch,0defh,0fedh,0cbah,0987h</p>
<p> dw 0,0,0,0,0,0,0,0,0,0</p>
<p>start: mov ax,<span style="text-decoration: underline"> codesg </span>;或mov ax,<span style="text-decoration: underline"> cs</span></p>
<p> mov ss,ax</p>
<p> mov sp,<span style="text-decoration: underline"> 24h </span> ;或mov sp,<span style="text-decoration: underline"> 36 </span> ;(第一版填1ah或26)</p>
<p> mov ax,0</p>
<p> mov ds,ax</p>
<p> mov bx,0</p>
<p> mov cx,8</p>
<p> s: push </p>
<p> <span style="text-decoration: underline"> pop cs: </span> ;或<span style="text-decoration: underline"> pop ss:</span></p>
<p> add bx,2 </p>
<p> loop s</p>
<p> mov ax,4c00h</p>
<p> int 21h</p>
<p>codesg ends</p>
<p>end start</p>
<p> </p>
<p> </p>
<p>C:\DOCUME~1\ADMINI~1>debug jc6-1-2.exe</p>
<p>-u</p>
<p>0C86:0024 B8860C MOV AX,0C86</p>
<p>0C86:0027 8ED0 MOV SS,AX</p>
<p>0C86:0029 BC2400 MOV SP,0024</p>
<p>0C86:002C B80000 MOV AX,0000</p>
<p>0C86:002F 8ED8 MOV DS,AX</p>
<p>0C86:0031 BB0000 MOV BX,0000</p>
<p>0C86:0034 B90800 MOV CX,0008</p>
<p>0C86:0037 FF37 PUSH </p>
<p>0C86:0039 2E CS:</p>
<p>0C86:003A 8F07 POP </p>
<p>0C86:003C 83C302 ADD BX,+02</p>
<p>0C86:003F E2F6 LOOP 0037</p>
<p>0C86:0041 B8004C MOV AX,4C00</p>
<p>-g 0041</p>
<p>AX=0000 BX=0010 CX=0000 DX=0000 SP=0024 BP=0000 SI=0000 DI=0000</p>
<p>DS=0000 ES=0C76 SS=0C86 CS=0C86 IP=0041 NV UP EI PL NZ AC PO NC</p>
<p>0C86:0041 B8004C MOV AX,4C00</p>
<p>-d 0:0 f</p>
<p>0000:0000 68 10 A7 00 BB 13 8D 06-16 00 B2 03 B1 13 8D 06 h...............</p>
<p>-d 0c86:0 f</p>
<p>0C86:0000 68 10 A7 00 BB 13 8D 06-16 00 B2 03 B1 13 8D 06 h...............</p>
<p>-q</p>
<p> </p>
<h3>第九章 转移指令的原理</h3>
<p>检测点9.1</p>
<p>(1)程序如下。</p>
<p>assume cs:code</p>
<p>data segment</p>
<p> dw 2 dup (0)</p>
<p>data ends</p>
<p>code segment</p>
<p> start: mov ax,dtat</p>
<p> mov ds,ax</p>
<p> mov bx,0</p>
<p> jmp word ptr </p>
<p>code ends</p>
<p>end start</p>
<p>若要使jmp指令执行后,CS:IP指向程序的第一条指令,在data段中应该定义哪些数据?</p>
<p> </p>
<p>答案①db 3 dup (0)</p>
<p>答案②dw 2 dup (0)</p>
<p>答案③dd 0</p>
<p>jmp word ptr 为段内转移,要CS:IP指向程序的第一条指令,应设置ds:的字单元(2个字节)存放数据应为0,则(ip)=ds:=0</p>
<p>简单来说就是,只要ds:起始地址的两个字节为0就可以了</p>
<p> </p>
<p>检测点9.1</p>
<p>(1)程序如下。</p>
<p>assume cs:code</p>
<p>data segment</p>
<p> dd 12345678h</p>
<p>data ends</p>
<p>code segment</p>
<p> start: mov ax,data</p>
<p> mov ds,ax</p>
<p> mov bx,0</p>
<p> mov ,<span style="text-decoration: underline"> bx </span> ;或mov ,<span style="text-decoration: underline"> word ptr 0 </span> ;或mov ,<span style="text-decoration: underline"> offset start</span></p>
<p> mov ,<span style="text-decoration: underline"> cs </span> ;或mov ,<span style="text-decoration: underline"> cs </span> ;或mov ,<span style="text-decoration: underline"> seg code </span></p>
<p> jmp dword ptr ds:</p>
<p>code ends</p>
<p>end start</p>
<p>补全程序,使用jmp指令执行后,CS:IP指向程序的第一条指令。</p>
<p> </p>
<p>第一格可填①mov ,bx ②mov ,word ptr 0 ③mov ,offset start等。</p>
<p>第二格可填①mov ,cs ②mov ,cs ③mov ,seg code等。</p>
<p>解析:</p>
<p>jmp dword ptr ds:为段间转移,(cs)=(内存单元地址+2),(ip)=(内存单元地址),要CS:IP指向程序的第一条指令,第一条程序地址cs:0,应设置CS:IP指向cs:0</p>
<p>程序中的mov ,bx这条指令,是将ip设置为0 </p>
<p>mov ,cs,将cs这个段地址放入内存单元 </p>
<p>执行后,cs应该不变,只调整ip为0,(ip)=ds:=0</p>
<p> </p>
<p>C:\DOCUME~1\SNUSER>debug jc9-1.exe </p>
<p>-r </p>
<p>AX=0000 BX=0000 CX=0021 DX=0000 SP=0000 BP=0000 SI=0000 DI=0000 </p>
<p>DS=0C3E ES=0C3E SS=0C4E CS=0C4F IP=0000 NV UP EI PL NZ NA PO NC </p>
<p>0C4F:0000 B84E0C MOV AX,0C4E </p>
<p>-t </p>
<p>AX=0C4E BX=0000 CX=0021 DX=0000 SP=0000 BP=0000 SI=0000 DI=0000 </p>
<p>DS=0C3E ES=0C3E SS=0C4E CS=0C4F IP=0003 NV UP EI PL NZ NA PO NC </p>
<p>0C4F:0003 8ED8 MOV DS,AX </p>
<p>-t </p>
<p>AX=0C4E BX=0000 CX=0021 DX=0000 SP=0000 BP=0000 SI=0000 DI=0000 </p>
<p>DS=0C4E ES=0C3E SS=0C4E CS=0C4F IP=0005 NV UP EI PL NZ NA PO NC </p>
<p>0C4F:0005 BB0000 MOV BX,0000 </p>
<p>-t </p>
<p>AX=0C4E BX=0000 CX=0021 DX=0000 SP=0000 BP=0000 SI=0000 DI=0000 </p>
<p>DS=0C4E ES=0C3E SS=0C4E CS=0C4F IP=0008 NV UP EI PL NZ NA PO NC </p>
<p>0C4F:0008 891F MOV ,BX DS:0000=5678 </p>
<p>-t </p>
<p>AX=0C4E BX=0000 CX=0021 DX=0000 SP=0000 BP=0000 SI=0000 DI=0000 </p>
<p>DS=0C4E ES=0C3E SS=0C4E CS=0C4F IP=000A NV UP EI PL NZ NA PO NC </p>
<p>0C4F:000A 8C4F02 MOV ,CS DS:0002=1234 </p>
<p>-t </p>
<p>AX=0C4E BX=0000 CX=0021 DX=0000 SP=0000 BP=0000 SI=0000 DI=0000 </p>
<p>DS=0C4E ES=0C3E SS=0C4E CS=0C4F IP=000D NV UP EI PL NZ NA PO NC </p>
<p>0C4F:000D FF2E0000 JMP FAR DS:0000=0000 </p>
<p>-t </p>
<p>AX=0C4E BX=0000 CX=0021 DX=0000 SP=0000 BP=0000 SI=0000 DI=0000 </p>
<p>DS=0C4E ES=0C3E SS=0C4E CS=0C4F IP=0000 NV UP EI PL NZ NA PO NC </p>
<p>0C4F:0000 B84E0C MOV AX,0C4E </p>
<p>-q</p>
<p>检测点9.1</p>
<p>(3)用Debug查看内存,结果如下:</p>
<p>2000:1000 BE 00 06 00 00 00 ......</p>
<p>则此时,CPU执行指令:</p>
<p>mov ax,2000h</p>
<p>mov es,ax</p>
<p>jmp dword ptr es:</p>
<p>后,(cs)=<span style="text-decoration: underline"> 0006H </span>,(ip)=<span style="text-decoration: underline"> 00BEH</span></p>
<p> </p>
<p>解析:</p>
<p>jmp dword ptr为段间转移,高位存放段地址,低位存放偏移地址</p>
<p>(cs)=(内存单元地址+2),(ip)=(内存单元地址)</p>
<p> </p>
<p>根据书P16,对于寄存器AX,AH为高位(前1字节为高位),AL为低位(后1字节为低位)</p>
<p>推算出(内存单元地址)=00beh,(内存单元地址+2)=0006h</p>
<p>根据书P182,高位存放段地址(后2个字节为高位),低位存放偏移地址(前2个字节为低位)</p>
<p>(cs)=(内存单元地址+2),(ip)=(内存单元地址)</p>
<p>推算出(cs)=0006h,(ip)=00beh</p>
<p> </p>
<p>用debug跟踪,可能会出现如下错误,debug给出的答案是(cs)不变,(ip)=1000h</p>
<p>C:\DOCUME~1\SNUSER>debug</p>
<p>-r es</p>
<p>ES 0BF9</p>
<p>:2000</p>
<p>-e 2000:1000 be 00 06 00 00 00</p>
<p>-a</p>
<p>0BF9:0100 mov ax,2000</p>
<p>0BF9:0103 mov es,ax</p>
<p>0BF9:0105 jmp dword ptr es:</p>
<p> ^ Error</p>
<p>0BF9:0105 jmp dword ptr 2000:1000</p>
<p>0BF9:0108</p>
<p>-r</p>
<p>AX=0000 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0BF9 ES=2000 SS=0BF9 CS=0BF9 IP=0100 NV UP EI PL NZ NA PO NC</p>
<p>0BF9:0100 B80020 MOV AX,2000</p>
<p>-t</p>
<p>AX=2000 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0BF9 ES=2000 SS=0BF9 CS=0BF9 IP=0103 NV UP EI PL NZ NA PO NC</p>
<p>0BF9:0103 8EC0 MOV ES,AX</p>
<p>-t</p>
<p>AX=2000 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0BF9 ES=2000 SS=0BF9 CS=0BF9 IP=0105 NV UP EI PL NZ NA PO NC</p>
<p>0BF9:0105 E9F80E JMP 1000</p>
<p>-t</p>
<p>AX=2000 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0BF9 ES=2000 SS=0BF9 CS=0BF9 IP=1000 NV UP EI PL NZ NA PO NC</p>
<p>0BF9:1000 E475 IN AL,75</p>
<p>出现错误的原因是:</p>
<p>jmp dword ptr es:对应的debug下的指令并不是你给出的</p>
<p>jmp dword ptr 2000:这样的形式,可以看出,当你写出上述指令后,运行的时候其指令仅仅变成了jmp 1000,缺少了一个指定段地址的指令。</p>
<p> </p>
<p>我们可以写一个源程序模拟一下上面的这段程序</p>
<p>assume cs:codesg </p>
<p>data segment</p>
<p> db 0BEH,0,6,0,0,0</p>
<p>data ends</p>
<p>codesg segment </p>
<p>start:</p>
<p>mov ax,data</p>
<p>mov es,ax</p>
<p>jmp dword ptr es:</p>
<p>codesg ends </p>
<p>end start </p>
<p>上面这个程序,数据地址是程序分配的,不是指定的那个地址,但是,对于我们理解程序运行的整个过程没有影响。下面是debug的信息</p>
<p>-t</p>
<p>AX=1438 BX=0000 CX=001A DX=0000 SP=0000 BP=0000 SI=0000 DI=0000</p>
<p>DS=1428 ES=1428 SS=1438 CS=1439 IP=0003 NV UP EI PL NZ NA PO NC</p>
<p>1439:0003 8EC0 MOV ES,AX</p>
<p>-t</p>
<p>AX=1438 BX=0000 CX=001A DX=0000 SP=0000 BP=0000 SI=0000 DI=0000</p>
<p>DS=1428 ES=1438 SS=1438 CS=1439 IP=0005 NV UP EI PL NZ NA PO NC</p>
<p>1439:0005 26 ES:</p>
<p>1439:0006 FF2E0000 JMP FAR ES:0000=00BE</p>
<p>-d es:0 f</p>
<p>1438:0000 BE 00 06 00 00 00 00 00-00 00 00 00 00 00 00 00 ................</p>
<p>-t</p>
<p>AX=1438 BX=0000 CX=001A DX=0000 SP=0000 BP=0000 SI=0000 DI=0000</p>
<p>DS=1428 ES=1438 SS=1438 CS=0006 IP=00BE NV UP EI PL NZ NA PO NC</p>
<p>0006:00BE 00F0 ADD AL,DH</p>
<p>我们可以看到,源程序中jmp dword ptr es: 对应的debug下的汇编指令是</p>
<p>1439:0005 26 ES:</p>
<p>1439:0006 FF2E0000 JMP FAR </p>
<p>而不是仅仅的一个(JMP 地址)那样的形式,所以,你在debug下的操作本身就是不行的。</p>
<p>另外,此题目的检测目的就是将内存中的数据作为跳转的CS和IP的值来进行跳转。对于给定的一个地址A,A开始的一个字单元是IP,A+2开始的一个字段元是CS。也就是以A为其实地址的内存中,低字单元是IP,高字单元是CS。</p>
<p> </p>
<p>如非要在DEBUG中进行操作,可用以下方式:</p>
<p>-e 2000:1000 be 00 06 00 00 00</p>
<p>-a</p>
<p>139A:0100 mov ax,2000</p>
<p>139A:0103 mov es,ax</p>
<p>139A:0105 es:</p>
<p>139A:0106 jmp far </p>
<p>139A:010A</p>
<p>-u</p>
<p>139A:0100 B80020 MOV AX,2000</p>
<p>139A:0103 8EC0 MOV ES,AX</p>
<p>139A:0105 26 ES:</p>
<p>139A:0106 FF2E0010 JMP FAR </p>
<p>-r</p>
<p>AX=0000 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=139A ES=139A SS=139A CS=139A IP=0100 NV UP EI PL NZ NA PO NC</p>
<p>139A:0100 B80020 MOV AX,2000</p>
<p>-t</p>
<p>AX=2000 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=139A ES=139A SS=139A CS=139A IP=0103 NV UP EI PL NZ NA PO NC</p>
<p>139A:0103 8EC0 MOV ES,AX</p>
<p>-t</p>
<p>AX=2000 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=139A ES=2000 SS=139A CS=139A IP=0105 NV UP EI PL NZ NA PO NC</p>
<p>139A:0105 26 ES:</p>
<p>139A:0106 FF2E0010 JMP FAR ES:1000=00BE</p>
<p>-t</p>
<p>AX=2000 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=139A ES=2000 SS=139A CS=0006 IP=00BE NV UP EI PL NZ NA PO NC</p>
<p>0006:00BE 00F0 ADD AL,DH</p>
<p>-</p>
<p> </p>
<p>检测点9.2</p>
<p>补全编程,利用jcxz指令,实现在内存2000H段中查找第一个值为0的字节,找到后,将它的偏移地址存储在dx中。</p>
<p>assume cs:code</p>
<p>code segment</p>
<p> start: mov ax,2000h</p>
<p> mov ds,ax</p>
<p> mov bx,0</p>
<p> s:<span style="text-decoration: underline"> mov ch,0 </span></p>
<p> <span style="text-decoration: underline"> mov cl,</span></p>
<p> <span style="text-decoration: underline"> jcxz ok </span> ;当cx=0时,CS:IP指向OK</p>
<p> <span style="text-decoration: underline"> inc bx </span></p>
<p> jmp short s</p>
<p> ok: mov dx,bx</p>
<p> mov ax ,4c00h</p>
<p> int 21h</p>
<p>code ends</p>
<p>end start</p>
<p> </p>
<p>检测点9.3</p>
<p>补全编程,利用loop指令,实现在内存2000H段中查找第一个值为0的字节,找到后,将它的偏移地址存储在dx中。</p>
<p>assume cs:code</p>
<p>code segment</p>
<p>start: mov ax,2000h</p>
<p> mov ds,ax</p>
<p> mov bx,0</p>
<p> s:mov cl,</p>
<p> mov ch,0</p>
<p> <span style="text-decoration: underline"> inc cx </span></p>
<p> inc bx</p>
<p> loop s</p>
<p> ok:dec bx</p>
<p> mov dx,bx</p>
<p> mov ax,4c00h</p>
<p> int 21h</p>
<p>code ends</p>
<p>end start</p>
<p> </p>
<p> </p>
<p>书P101,执行loop s时,首先要将(cx)减1。</p>
<p>“loop 标号”相当于</p>
<p>dec cx</p>
<p>if((cx)≠0) jmp short 标号</p>
<p> </p>
<h3>第十章 CALL和RET指令</h3>
<p>检测点10.1</p>
<p>补全程序,实现从内存1000:0000处开始执行指令。</p>
<p>assume cs:code</p>
<p>stack segment</p>
<p> db 16 dup (0)</p>
<p>stack ends</p>
<p>code segment</p>
<p>start: mov ax,stack</p>
<p> mov ss,ax</p>
<p> mov sp,16</p>
<p> mov ax,<span style="text-decoration: underline"> 1000h</span></p>
<p> push ax</p>
<p> mov ax,<span style="text-decoration: underline"> 0 </span></p>
<p> push ax</p>
<p> retf</p>
<p>code ends</p>
<p>end start</p>
<p> </p>
<p> </p>
<p>执行reft指令时,相当于进行:</p>
<p>pop ip</p>
<p>pop cs</p>
<p>根据栈先进后出原则,应先将段地址cs入栈,再将偏移地址ip入栈。</p>
<p> </p>
<p> </p>
<p>C:\DOCUME~1\SNUSER>debug jc10-1.exe</p>
<p>-u</p>
<p>0C50:0000 B84F0C MOV AX,0C4F</p>
<p>0C50:0003 8ED0 MOV SS,AX</p>
<p>0C50:0005 BC1000 MOV SP,0010</p>
<p>0C50:0008 B80010 MOV AX,1000</p>
<p>0C50:000B 50 PUSH AX</p>
<p>0C50:000C B80000 MOV AX,0000</p>
<p>0C50:000F 50 PUSH AX</p>
<p>0C50:0010 CB RETF</p>
<p>0C50:0011 3986FEFE CMP ,AX</p>
<p>0C50:0015 737D JNB 0094</p>
<p>-g 0010</p>
<p>AX=0000 BX=0000 CX=0021 DX=0000 SP=000C BP=0000 SI=0000 DI=0000</p>
<p>DS=0C3F ES=0C3F SS=0C4F CS=0C50 IP=0010 NV UP EI PL NZ NA PO NC</p>
<p>0C50:0010 CB RETF</p>
<p>-t</p>
<p>AX=0000 BX=0000 CX=0021 DX=0000 SP=0010 BP=0000 SI=0000 DI=0000</p>
<p>DS=0C3F ES=0C3F SS=0C4F CS=1000 IP=0000 NV UP EI PL NZ NA PO NC</p>
<p>1000:0000 6E DB 6E</p>
<p>-</p>
<p> </p>
<p>检测点10.2</p>
<p>下面的程序执行后,ax中的数值为多少?</p>
<p>内存地址 机器码 汇编指令 执行后情况</p>
<p>1000:0 b8 00 00 mov ax,0 ax=0 ip指向1000:3</p>
<p>1000:3 e8 01 00 call s pop ip ip指向1000:7</p>
<p>1000:6 40 inc ax</p>
<p>1000:7 58 s:pop ax ax=6</p>
<p> </p>
<p>用debug进行跟踪确认,“call 标号”是将该指令后的第一个字节偏移地址入栈,再转到标号处执行指令。</p>
<p> </p>
<p>assume cs:code</p>
<p>code segment</p>
<p>start: mov ax,0</p>
<p> call s</p>
<p> inc ax</p>
<p>s: pop ax</p>
<p> mov ax,4c00h</p>
<p> int 21h</p>
<p>code ends</p>
<p>end start</p>
<p> </p>
<p>C:\DOCUME~1\SNUSER>debug jc10-2.exe</p>
<p>-u</p>
<p>0C4F:0000 B80000 MOV AX,0000</p>
<p>0C4F:0003 E80100 CALL 0007</p>
<p>0C4F:0006 40 INC AX</p>
<p>0C4F:0007 58 POP AX</p>
<p>0C4F:0008 B8004C MOV AX,4C00</p>
<p>0C4F:000B CD21 INT 21</p>
<p>-r</p>
<p>AX=0000 BX=0000 CX=000D DX=0000 SP=0000 BP=0000 SI=0000 DI=0000</p>
<p>DS=0C3F ES=0C3F SS=0C4F CS=0C4F IP=0000 NV UP EI PL NZ NA PO NC</p>
<p>0C4F:0000 B80000 MOV AX,0000</p>
<p>-t</p>
<p>AX=0000 BX=0000 CX=000D DX=0000 SP=0000 BP=0000 SI=0000 DI=0000</p>
<p>DS=0C3F ES=0C3F SS=0C4F CS=0C4F IP=0003 NV UP EI PL NZ NA PO NC</p>
<p>0C4F:0003 E80100 CALL 0007</p>
<p>-t</p>
<p>AX=0000 BX=0000 CX=000D DX=0000 SP=FFFE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C3F ES=0C3F SS=0C4F CS=0C4F IP=0007 NV UP EI PL NZ NA PO NC</p>
<p>0C4F:0007 58 POP AX</p>
<p>-t</p>
<p>AX=0006 BX=0000 CX=000D DX=0000 SP=0000 BP=0000 SI=0000 DI=0000</p>
<p>DS=0C3F ES=0C3F SS=0C4F CS=0C4F IP=0008 NV UP EI PL NZ NA PO NC</p>
<p>0C4F:0008 B8004C MOV AX,4C00</p>
<p>-t</p>
<p>AX=4C00 BX=0000 CX=000D DX=0000 SP=0000 BP=0000 SI=0000 DI=0000</p>
<p>DS=0C3F ES=0C3F SS=0C4F CS=0C4F IP=000B NV UP EI PL NZ NA PO NC</p>
<p>0C4F:000B CD21 INT 21</p>
<p>-p</p>
<p>Program terminated normally</p>
<p> </p>
<p>检测点10.3</p>
<p>下面的程序执行后,ax中的数值为多少?</p>
<p>内存地址 机器码 汇编指令 执行后情况</p>
<p>1000:0 b8 00 00 mov ax,0 ax=0,ip指向1000:3</p>
<p>1000:3 9a 09 00 00 10 call far ptr s pop cs,pop ip,ip指向1000:9</p>
<p>1000:8 40 inc ax</p>
<p>1000:9 58 s:pop ax ax=8h</p>
<p> add ax,ax ax=10h</p>
<p> pop bx bx=1000h</p>
<p> add ax,bx ax=1010h</p>
<p> </p>
<p>用debug进行跟踪确认,“call far ptr s”是先将该指令后的第一个字节段地址cs=1000h入栈,再将偏移地址ip=8h入栈,最后转到标号处执行指令。</p>
<p>出栈时,根据栈先进后出的原则,先出的为ip=8h,后出的为cs=1000h</p>
<p> </p>
<p>检测点10.4</p>
<p>下面的程序执行后,ax中的数值为多少?</p>
<p>内存地址 机器码 汇编指令 执行后情况</p>
<p>1000:0 b8 06 00 mov ax,6 ax=6,ip指向1000:3</p>
<p>1000:3 ff d0 call ax pop ip,ip指向1000:6</p>
<p>1000:5 40 inc ax</p>
<p>1000:6 58 mov bp,sp bp=sp=fffeh</p>
<p> add ax, ax==6+5=0bh</p>
<p> </p>
<p>用debug进行跟踪确认,“call ax(16位reg)”是先将该指令后的第一个字节偏移地址ip入栈,再转到偏移地址为ax(16位reg)处执行指令。</p>
<p> </p>
<p>检测点10.5</p>
<p>(1)下面的程序执行后,ax中的数值为多少?</p>
<p>assume cs:code</p>
<p>stack segment</p>
<p> dw 8 dup (0)</p>
<p>stack ends</p>
<p>code segment</p>
<p>start: mov ax,stack</p>
<p> mov ss,ax</p>
<p> mov sp,16</p>
<p> mov ds,ax</p>
<p> mov ax,0</p>
<p> call word ptr ds:</p>
<p> inc ax</p>
<p> inc ax</p>
<p> inc ax</p>
<p> mov ax,4c00h</p>
<p> int 21h</p>
<p>code ends</p>
<p>end start</p>
<p> </p>
<p>推算:</p>
<p>执行call word ptr ds:指令时,先cs入栈,再ip=11入栈,最后ip转移到(ds:)。(ds:)=11h,执行inc ax……最终ax=3</p>
<p> </p>
<p>题中特别关照别用debug跟踪,跟踪结果不一定正确,但还是忍不住去试试,看是什么结果。</p>
<p>根据单步跟踪发现,执行call word ptr ds:指令时,显示ds:=065D。</p>
<p>ds:0000~ds:0010不是已设置成stack数据段了嘛,不是应该全都是0的嘛。</p>
<p>于是进行了更详细的单步跟踪,发现初始数据段中数据确实为0,但执行完mov ss,ax;mov sp,16这两条指令后,数据段中数据发生改变。这是为什么呢?中断呗~~~~</p>
<p> </p>
<p>C:\DOCUME~1\SNUSER>debug jc10-5.exe</p>
<p>-u</p>
<p>0C50:0000 B84F0C MOV AX,0C4F</p>
<p>0C50:0003 8ED0 MOV SS,AX</p>
<p>0C50:0005 BC1000 MOV SP,0010</p>
<p>0C50:0008 8ED8 MOV DS,AX</p>
<p>0C50:000A B80000 MOV AX,0000</p>
<p>0C50:000D FF160E00 CALL </p>
<p>0C50:0011 40 INC AX</p>
<p>0C50:0012 40 INC AX</p>
<p>0C50:0013 40 INC AX</p>
<p>0C50:0014 B8004C MOV AX,4C00</p>
<p>0C50:0017 CD21 INT 21</p>
<p>-r</p>
<p>AX=0000 BX=0000 CX=0029 DX=0000 SP=0000 BP=0000 SI=0000 DI=0000</p>
<p>DS=0C3F ES=0C3F SS=0C4F CS=0C50 IP=0000 NV UP EI PL NZ NA PO NC</p>
<p>0C50:0000 B84F0C MOV AX,0C4F</p>
<p>-d 0c4f:0 f</p>
<p>0C4F:0000 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 ................</p>
<p>-t</p>
<p>AX=0C4F BX=0000 CX=0029 DX=0000 SP=0000 BP=0000 SI=0000 DI=0000</p>
<p>DS=0C3F ES=0C3F SS=0C4F CS=0C50 IP=0003 NV UP EI PL NZ NA PO NC</p>
<p>0C50:0003 8ED0 MOV SS,AX</p>
<p>-d 0c4f:0 f</p>
<p>0C4F:0000 00 00 00 00 00 00 00 00-00 00 00 00 00 00 00 00 ................</p>
<p>-t</p>
<p>AX=0C4F BX=0000 CX=0029 DX=0000 SP=0010 BP=0000 SI=0000 DI=0000</p>
<p>DS=0C3F ES=0C3F SS=0C4F CS=0C50 IP=0008 NV UP EI PL NZ NA PO NC</p>
<p>0C50:0008 8ED8 MOV DS,AX</p>
<p>-d 0c4f:0 f</p>
<p>0C4F:0000 00 00 00 00 00 00 4F 0C-00 00 08 00 50 0C 5D 06 ......O.....P.].</p>
<p>-t</p>
<p>AX=0C4F BX=0000 CX=0029 DX=0000 SP=0010 BP=0000 SI=0000 DI=0000</p>
<p>DS=0C4F ES=0C3F SS=0C4F CS=0C50 IP=000A NV UP EI PL NZ NA PO NC</p>
<p>0C50:000A B80000 MOV AX,0000</p>
<p>-d 0c4f:0 f</p>
<p>0C4F:0000 00 00 00 00 00 00 4F 0C-00 00 0A 00 50 0C 5D 06 ......O.....P.].</p>
<p>-t</p>
<p>AX=0000 BX=0000 CX=0029 DX=0000 SP=0010 BP=0000 SI=0000 DI=0000</p>
<p>DS=0C4F ES=0C3F SS=0C4F CS=0C50 IP=000D NV UP EI PL NZ NA PO NC</p>
<p>0C50:000D FF160E00 CALL DS:000E=065D</p>
<p>-d 0c4f:0 f</p>
<p>0C4F:0000 00 00 00 00 00 00 00 00-00 00 0D 00 50 0C 5D 06 ............P.].</p>
<p>-</p>
<p> </p>
<p>检测点10.5</p>
<p>(2)下面的程序执行后,ax和bx中的数值为多少?</p>
<p>assume cs:codesg</p>
<p>stack segment</p>
<p> dw 8 dup(0)</p>
<p>stack ends</p>
<p>codesg segment</p>
<p>start:</p>
<p> mov ax,stack</p>
<p> mov ss,ax</p>
<p> mov sp,10h</p>
<p> mov word ptr ss:,offset s ;(ss:)=1ah</p>
<p> mov ss:,cs ;(ss:)=cs</p>
<p> call dword ptr ss: ;cs入栈,ip=19h入栈,转到cs:1ah处执行指令</p>
<p> ;(ss:)=cs,(ss:)=ip</p>
<p> nop</p>
<p>s: mov ax,offset s ;ax=1ah</p>
<p> sub ax,ss: ;ax=1ah-(ss:)=1ah-19h=1</p>
<p> mov bx,cs ;bx=cs=0c5bh</p>
<p> sub bx,ss: ;bx=cs-cs=0</p>
<p> mov ax,4c00h</p>
<p> int 21h</p>
<p>codesg ends</p>
<p>end start</p>
<p> </p>
<p>C:\DOCUME~1\ADMINI~1>debug jc10-5.exe</p>
<p>-u</p>
<p>0C5B:0000 B85A0C MOV AX,0C5A</p>
<p>0C5B:0003 8ED0 MOV SS,AX</p>
<p>0C5B:0005 BC1000 MOV SP,0010</p>
<p>0C5B:0008 36 SS:</p>
<p>0C5B:0009 C70600001A00 MOV WORD PTR ,001A</p>
<p>0C5B:000F 36 SS:</p>
<p>0C5B:0010 8C0E0200 MOV ,CS</p>
<p>0C5B:0014 36 SS:</p>
<p>0C5B:0015 FF1E0000 CALL FAR </p>
<p>0C5B:0019 90 NOP</p>
<p>0C5B:001A B81A00 MOV AX,001A</p>
<p>0C5B:001D 36 SS:</p>
<p>0C5B:001E 2B060C00 SUB AX,</p>
<p>-u</p>
<p>0C5B:0022 8CCB MOV BX,CS</p>
<p>0C5B:0024 36 SS:</p>
<p>0C5B:0025 2B1E0E00 SUB BX,</p>
<p>0C5B:0029 B8004C MOV AX,4C00</p>
<p> </p>
<h3>第十一章 标志寄存器</h3>
<p>检测点11.1</p>
<p>写出下面每条指令执行后,ZF、PF、SF、等标志位的值。</p>
<p>sub al,al al=0h ZF=1 PF=1 SF=0 </p>
<p>mov al,1 al=1h ZF=1 PF=1 SF=0 </p>
<p>push ax ax=1h ZF=1 PF=1 SF=0 </p>
<p>pop bx bx=1h ZF=1 PF=1 SF=0 </p>
<p>add al,bl al=2h ZF=0 PF=0 SF=0 </p>
<p>add al,10 al=12h ZF=0 PF=1 SF=0 </p>
<p>mul al ax=144h ZF=0 PF=1 SF=0</p>
<p> </p>
<p>检测点涉及的相关内容:</p>
<p>ZF是flag的第6位,零标志位,记录指令执行后结果是否为0,结果为0时,ZF=1</p>
<p>PF是flag的第2位,奇偶标志位,记录指令执行后结果二进制中1的个数是否为偶数,结果为偶数时,PF=1</p>
<p>SF是flag的第7位,符号标志位,记录有符号运算结果是否为负数,结果为负数时,SF=1</p>
<p>add、sub、mul、div 、inc、or、and等运算指令影响标志寄存器</p>
<p>mov、push、pop等传送指令对标志寄存器没影响。</p>
<p> </p>
<p>C:\DOCUME~1\ADMINI~1>debug</p>
<p>-a</p>
<p>0C1C:0100 sub al,al</p>
<p>0C1C:0102 mov al,1</p>
<p>0C1C:0104 push ax</p>
<p>0C1C:0105 pop bx</p>
<p>0C1C:0106 add al,bl</p>
<p>0C1C:0108 add al,10</p>
<p>0C1C:010A mul al</p>
<p>0C1C:010C</p>
<p>-t</p>
<p>AX=0000 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=0102 NV UP EI PL ZR NA PE NC</p>
<p>0C1C:0102 B001 MOV AL,01</p>
<p>-t</p>
<p>AX=0001 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=0104 NV UP EI PL ZR NA PE NC</p>
<p>0C1C:0104 50 PUSH AX</p>
<p>-t</p>
<p>AX=0001 BX=0000 CX=0000 DX=0000 SP=FFEC BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=0105 NV UP EI PL ZR NA PE NC</p>
<p>0C1C:0105 5B POP BX</p>
<p>-t</p>
<p>AX=0001 BX=0001 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=0106 NV UP EI PL ZR NA PE NC</p>
<p>0C1C:0106 00D8 ADD AL,BL</p>
<p>-t</p>
<p>AX=0002 BX=0001 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=0108 NV UP EI PL NZ NA PO NC</p>
<p>0C1C:0108 0410 ADD AL,10</p>
<p>-t</p>
<p>AX=0012 BX=0001 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=010A NV UP EI PL NZ NA PE NC</p>
<p>0C1C:010A F6E0 MUL AL</p>
<p>-t</p>
<p>AX=0144 BX=0001 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0C1C ES=0C1C SS=0C1C CS=0C1C IP=010C OV UP EI PL NZ NA PE CY</p>
<p>0C1C:010C 1599CD ADC AX,CD99</p>
<p>-</p>
<p> </p>
<p>检测点11.2</p>
<p>写出下面每条指令执行后,ZF、PF、SF、CF、OF等标志位的值。</p>
<p> al CF OF SF ZF PF</p>
<p>sub al,al 0h/0000 0000b 0 0 0 1 1</p>
<p>mov al,10h 10h/0010 0000b 0 0 0 1 1</p>
<p>add al,90h a0h/1010 0000b 0 0 1 0 1</p>
<p>mov al,80h 80h/1000 0000b 0 0 1 0 1</p>
<p>add al,80h 0h/0000 0000b 1 1 0 1 1</p>
<p>mov al,0fch 0fch/1111 1100b 1 1 0 1 1</p>
<p>add al,05h 1h/0000 0001b 1 0 0 0 0</p>
<p>mov al,7dh 7dh/1111 1101b 1 0 0 0 0</p>
<p>add al,0bh 88h/1000 1000b 0 1 1 0 1</p>
<p> </p>
<p>检测点涉及的相关内容:</p>
<p>ZF是flag的第6位,零标志位,记录指令执行后结果是否为0,结果为0时,ZF=1 </p>
<p>PF是flag的第2位,奇偶标志位,记录指令执行后结果二进制数中1的个数是否为偶数,结果为偶数时,PF=1 </p>
<p>SF是flag的第7位,符号标志位,记录有符号运算结果是否为负数,结果为负数时,SF=1 </p>
<p>CF是flag的第0位,进位标志位,记录无符号运算结果是否有进/借位,结果有进/借位时,SF=1</p>
<p>OF是flag的第11位,溢出标志位,记录有符号运算结果是否溢出,结果溢出时,OF=1</p>
<p>add、sub、mul、div 、inc、or、and等运算指令影响flag</p>
<p>mov、push、pop等传送指令对flag没影响</p>
<p> </p>
<p>Microsoft(R) Windows DOS</p>
<p>(C)Copyright Microsoft Corp 1990-2001.</p>
<p>C:\DOCUME~1\SNUSER>debug</p>
<p>-a</p>
<p>0BF9:0100 sub al,al</p>
<p>0BF9:0102 mov al,10</p>
<p>0BF9:0104 add al,90</p>
<p>0BF9:0106 mov al,80</p>
<p>0BF9:0108 mov al,fc</p>
<p>0BF9:010A add al,5</p>
<p>0BF9:010C mov al,7d</p>
<p>0BF9:010E add al,b</p>
<p>0BF9:0110</p>
<p>-r</p>
<p>AX=0000 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0BF9 ES=0BF9 SS=0BF9 CS=0BF9 IP=0100 NV UP EI PL NZ NA PO NC</p>
<p>0BF9:0100 28C0 SUB AL,AL</p>
<p>-t</p>
<p>AX=0000 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0BF9 ES=0BF9 SS=0BF9 CS=0BF9 IP=0102 NV UP EI PL ZR NA PE NC</p>
<p>0BF9:0102 B010 MOV AL,10</p>
<p>-t</p>
<p>AX=0010 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0BF9 ES=0BF9 SS=0BF9 CS=0BF9 IP=0104 NV UP EI PL ZR NA PE NC</p>
<p>0BF9:0104 0490 ADD AL,90</p>
<p>-t</p>
<p>AX=00A0 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0BF9 ES=0BF9 SS=0BF9 CS=0BF9 IP=0106 NV UP EI NG NZ NA PE NC</p>
<p>0BF9:0106 B080 MOV AL,80</p>
<p>-t</p>
<p>AX=0080 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0BF9 ES=0BF9 SS=0BF9 CS=0BF9 IP=0108 NV UP EI NG NZ NA PE NC</p>
<p>0BF9:0108 B0FC MOV AL,FC</p>
<p>-t</p>
<p>AX=00FC BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0BF9 ES=0BF9 SS=0BF9 CS=0BF9 IP=010A NV UP EI NG NZ NA PE NC</p>
<p>0BF9:010A 0405 ADD AL,05</p>
<p>-t</p>
<p>AX=0001 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0BF9 ES=0BF9 SS=0BF9 CS=0BF9 IP=010C NV UP EI PL NZ AC PO CY</p>
<p>0BF9:010C B07D MOV AL,7D</p>
<p>-t</p>
<p>AX=007D BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0BF9 ES=0BF9 SS=0BF9 CS=0BF9 IP=010E NV UP EI PL NZ AC PO CY</p>
<p>0BF9:010E 040B ADD AL,0B</p>
<p>-t</p>
<p>AX=0088 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0BF9 ES=0BF9 SS=0BF9 CS=0BF9 IP=0110 OV UP EI NG NZ AC PE NC</p>
<p>0BF9:0110 C6BF1F9903 MOV BYTE PTR ,03 DS:991F=00</p>
<p>-</p>
<p> </p>
<p>检测点11.3</p>
<p>(1)补全下面的程序,统计F000:0处32个字节中,大小在的数据个数。</p>
<p> mov ax,0f000h</p>
<p> mov ds,ax</p>
<p> mov bx,0 ;ds:bx指向第一个字节</p>
<p> mov dx,0 ;初始化累加器</p>
<p> mov cx,32</p>
<p>s: mov al,</p>
<p> cmp al,32 ;和32进行比较</p>
<p> <span style="text-decoration: underline">jb s0 </span> ;如果低于al转到s0,继续循环</p>
<p> cmp al,128 ;和128进行比较</p>
<p> <span style="text-decoration: underline">ja s0 </span> ;如果高于al转到s0,继续循环</p>
<p> inc dx</p>
<p>s0: inc bx</p>
<p> loop s</p>
<p> </p>
<p>是闭区间,包括两端点的值</p>
<p>(32,128)是开区间,不包括两端点的值</p>
<p> </p>
<p>检测点11.3</p>
<p>(2)补全下面的程序,统计F000:0处32个字节中,大小在(32,128)的数据个数。</p>
<p> mov ax,0f000h</p>
<p> mov ds,ax</p>
<p> mov bx,0 ;ds:bx指向第一个字节</p>
<p> mov dx,0 ;初始化累加器</p>
<p> mov cx,32</p>
<p>s: mov al,</p>
<p> cmp al,32 ;和32进行比较</p>
<p> <span style="text-decoration: underline">jna s0 </span> ;如果不高于al转到s0,继续循环</p>
<p> cmp al,128 ;和128进行比较</p>
<p> <span style="text-decoration: underline">jnb s0 </span> ;如果不低于al转到s0,继续循环</p>
<p> inc dx</p>
<p>s0: inc bx</p>
<p> loop s</p>
<p> </p>
<p>是闭区间,包括两端点的值</p>
<p>(32,128)是开区间,不包括两端点的值</p>
<p> </p>
<p>检测点11.4</p>
<p>下面指令执行后,(ax)=<span style="text-decoration: underline"> 45h</span></p>
<p>mov ax,0</p>
<p>push ax</p>
<p>popf</p>
<p>mov ax,0fff0h</p>
<p>add ax,0010h</p>
<p>pushf</p>
<p>pop ax</p>
<p>and al,11000101B</p>
<p>and ah,00001000B</p>
<p> </p>
<p>推算过程:</p>
<p>popf后,标志寄存器中,本章节介绍的那些标志位都为0(但是此时标志寄存器并不是所有位置都为0,这个不用关心,没学过的位置用*先代替),向下进行,那么pushf将计算后的当时状态的标志寄存器入栈,然后pop给ax,这是ax是寄存器的值(这个值中包含了我们的*号),接下来就是对那些没有学过的标志位的屏蔽操作,这就是最后两条指令的意义所在,将不确定的位置都归0,那么只剩下我们能够确定的位置了,所以,结果就可以推理出来了。</p>
<p>mov ax,0 </p>
<p>push ax </p>
<p>popf </p>
<p>mov ax,0fff0h </p>
<p>add ax,0010h </p>
<p>pushf</p>
<p>pop ax 0 0 0 0 of df if tf sf zf 0 af 0 pf 0 cf</p>
<p> 0 0 0 0 0 0 * * 0 1 0 * 0 1 0 1</p>
<p> ax=flag=000000** 010*0101b</p>
<p>and al,11000101B al=01000101b=45h</p>
<p>and ah,00001000B ah=00000000b=0h</p>
<p> </p>
<p> </p>
<p>C:\DOCUME~1\SNUSER>debug</p>
<p>-a</p>
<p>0BF9:0100 mov ax,0</p>
<p>0BF9:0103 push ax</p>
<p>0BF9:0104 popf</p>
<p>0BF9:0105 mov ax,fff0</p>
<p>0BF9:0108 add ax,10</p>
<p>0BF9:010B pushf</p>
<p>0BF9:010C pop ax</p>
<p>0BF9:010D and al,c5</p>
<p>0BF9:010F and ah,8</p>
<p>0BF9:0112</p>
<p>-r</p>
<p>AX=0000 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0BF9 ES=0BF9 SS=0BF9 CS=0BF9 IP=0100 NV UP EI PL NZ NA PO NC</p>
<p>0BF9:0100 B80000 MOV AX,0000</p>
<p>-t</p>
<p>AX=0000 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0BF9 ES=0BF9 SS=0BF9 CS=0BF9 IP=0103 NV UP EI PL NZ NA PO NC</p>
<p>0BF9:0103 50 PUSH AX</p>
<p>-t</p>
<p>AX=0000 BX=0000 CX=0000 DX=0000 SP=FFEC BP=0000 SI=0000 DI=0000</p>
<p>DS=0BF9 ES=0BF9 SS=0BF9 CS=0BF9 IP=0104 NV UP EI PL NZ NA PO NC</p>
<p>0BF9:0104 9D POPF</p>
<p>-t</p>
<p>AX=0000 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0BF9 ES=0BF9 SS=0BF9 CS=0BF9 IP=0105 NV UP DI PL NZ NA PO NC</p>
<p>0BF9:0105 B8F0FF MOV AX,FFF0</p>
<p>-t</p>
<p>AX=FFF0 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0BF9 ES=0BF9 SS=0BF9 CS=0BF9 IP=0108 NV UP DI PL NZ NA PO NC</p>
<p>0BF9:0108 051000 ADD AX,0010</p>
<p>-t</p>
<p>AX=0000 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0BF9 ES=0BF9 SS=0BF9 CS=0BF9 IP=010B NV UP DI PL ZR NA PE CY</p>
<p>0BF9:010B 9C PUSHF</p>
<p>-t</p>
<p>AX=0000 BX=0000 CX=0000 DX=0000 SP=FFEC BP=0000 SI=0000 DI=0000</p>
<p>DS=0BF9 ES=0BF9 SS=0BF9 CS=0BF9 IP=010C NV UP DI PL ZR NA PE CY</p>
<p>0BF9:010C 58 POP AX</p>
<p>-t</p>
<p>AX=3047 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0BF9 ES=0BF9 SS=0BF9 CS=0BF9 IP=010D NV UP DI PL ZR NA PE CY</p>
<p>0BF9:010D 24C5 AND AL,C5</p>
<p>-t</p>
<p>AX=3045 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0BF9 ES=0BF9 SS=0BF9 CS=0BF9 IP=010F NV UP DI PL NZ NA PO NC</p>
<p>0BF9:010F 80E408 AND AH,08</p>
<p>-t</p>
<p>AX=0045 BX=0000 CX=0000 DX=0000 SP=FFEE BP=0000 SI=0000 DI=0000</p>
<p>DS=0BF9 ES=0BF9 SS=0BF9 CS=0BF9 IP=0112 NV UP DI PL ZR NA PE NC</p>
<p>0BF9:0112 4C DEC SP</p>
<p> </p>
<h3>第十二章 内中断</h3>
<p>检测点12.1</p>
<p>(1)用debug查看内存,情况如下:</p>
<p>0000:0000 68 10 A7 00 8B 01 70 00-16 00 9D 03 8B 01 70 00</p>
<p>则3号中断源对应的中断处理程序入口的偏移地址的内存单位的地址为:<span style="text-decoration: underline"> 0070:018b</span></p>
<p>检测点涉及相关内容:</p>
<p>一个表项存放一个中断向量,也就是一个中断处理程序的入口地址,这个入口地址包括段地址和偏移地址,一个表项占两个字,高地址存放段地址,低地址存放偏移地址</p>
<p> </p>
<p>检测点12.1</p>
<p>(2)</p>
<p>存储N号中断源对应的中断处理程序入口的偏移地址的内存单元的地址为:<span style="text-decoration: underline"> 4N</span></p>
<p>存储N号中断源对应的中断处理程序入口的段地址的内存单元的地址为:<span style="text-decoration: underline"> 4N+2</span></p>
<p>检测点涉及相关内容:</p>
<p>一个表项存放一个中断向量,也就是一个中断处理程序的入口地址,这个入口地址包括段地址和偏移地址,一个表项占两个字,高地址存放段地址,低地址存放偏移地址</p>
<p> </p>
<h3> </h3>
<h3>第十三章 int指令</h3>
<p>检测点13.1</p>
<p> </p>
<p>7ch中断例程如下:</p>
<p>lp: push bp</p>
<p> mov bp,sp</p>
<p> dec cx</p>
<p> jcxz lpret</p>
<p> add ,bx</p>
<p>lpret: pop bp</p>
<p> iret</p>
<p> (1)在上面的内容中,我们用7ch中断例程实现loop的功能,则上面的7ch中断例程所能进行的最大转移位移是多少?</p>
<p>最大位移是FFFFH</p>
<p> </p>
<p>检测点13.1</p>
<p>(2)用7ch中断例程完成jmp near ptr s指令功能,用bx向中断例程传送转移位移。</p>
<p> </p>
<p>应用举例:在屏幕的第12行,显示data段中以0结尾的字符串。</p>
<p>assume cs:code</p>
<p>data segment</p>
<p> db 'conversation',0</p>
<p>data ends</p>
<p>code segment</p>
<p>start:</p>
<p> mov ax,data</p>
<p> mov ds,ax</p>
<p> mov si,0</p>
<p> mov ax,0b800h</p>
<p> mov es,ax</p>
<p> mov di,12*160</p>
<p>s: cmp byte ptr ,0</p>
<p> je ok</p>
<p> mov al,</p>
<p> mov es:,al</p>
<p> inc si</p>
<p> add di,2</p>
<p> mov bx,offset s-offset ok</p>
<p> int 7ch</p>
<p>ok: mov ax,4c00h</p>
<p> int 21h</p>
<p>code ends</p>
<p>end start</p>
<p> </p>
<p>jmp near ptr s指令的功能为:(ip)=(ip)+16位移,实现段内近转移</p>
<p> </p>
<p>assume cs:code </p>
<p>code segment </p>
<p>start:</p>
<p>mov ax,cs </p>
<p>mov ds,ax </p>
<p>mov si,offset do0 ;设置ds:si指向源地址 </p>
<p>mov ax,0 </p>
<p>mov es,ax </p>
<p>mov di,200h ;设置es:di指向目标地址 </p>
<p>mov cx,offset do0end-offset do0 ;设置cx为传输长度 </p>
<p>cld ;设置传输方向为正 </p>
<p>rep movsb </p>
<p>mov ax,0 </p>
<p>mov es,ax </p>
<p>mov word ptr es:,200h </p>
<p>mov word ptr es:,0 ;设置中断向量表 </p>
<p>mov ax,4c00h </p>
<p>int 21h </p>
<p>do0:</p>
<p> push bp</p>
<p>mov bp,sp</p>
<p> add ,bx ;ok的偏移地址+bx得到s的偏移地址</p>
<p>pop bp</p>
<p>iret</p>
<p>mov ax,4c00h </p>
<p>int 21h </p>
<p>do0end:</p>
<p> nop</p>
<p>code ends</p>
<p>end start</p>
<p> </p>
<p>检测点13.2</p>
<p>判断下面说法的正误:</p>
<p>(1)我们可以编程改变FFFF:0处的指令,使得CPU不去执行BIOS中的硬件系统检测和初始化程序。</p>
<p> </p>
<p>答:错误,FFFF:0处的内容无法改变。</p>
<p> </p>
<p> </p>
<h3>第十四章 端口</h3>
<p>检测点14.1 读取写入CMOS RAM单元内容</p>
<p>(1)编程,读取CMOS RAM的2号单元内容。</p>
<p> </p>
<p>assume cs:code</p>
<p>code segment</p>
<p>start: mov al,2 ;赋值al</p>
<p> out 70h,al ;将al送入端口70h</p>
<p> in al,71h ;从端口71h处读出单元内容</p>
<p> mov ax,4c00h</p>
<p> int 21h</p>
<p>code ends</p>
<p>end start</p>
<p> </p>
<p>检测点14.1</p>
<p>(2)编程,向CMOS RAM的2号单元写入0。</p>
<p> </p>
<p>assume cs:code</p>
<p>code segment</p>
<p>start: mov al,2 ;赋值al</p>
<p> out 70h,al ;将al送入端口70h</p>
<p> mov al,0 ;赋值al</p>
<p> out 71h,al ;向端口71h写入数据al</p>
<p> mov ax,4c00h</p>
<p> int 21h</p>
<p>code ends</p>
<p>end start</p>
<p> </p>
<p>检测点14.2 用加法和移位指令计算</p>
<p><img src="https://blog.csdn.net/qq_42777804/article/details/%E6%A3%80%E6%B5%8B%E7%82%B914.2.files/image001.jpg" alt="" class="has"></p>
<p>效果图</p>
<p><img src="https://img-blog.csdnimg.cn/20190527231412247.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzQyNzc3ODA0,size_16,color_FFFFFF,t_70" alt="" width="479" height="241" class="has"></p>
<p>编程,用加法和移位指令计算(ax)=(ax)*10</p>
<p>提示:(ax)*10=(ax)*2+(ax)*8</p>
<p> </p>
<p>assume cs:code</p>
<p>code segment</p>
<p>start: mov bx,ax</p>
<p> shl ax,1 ;左移1位(ax)=(ax)*2</p>
<p> mov cl,3</p>
<p> shl bx,cl ;左移3位(bx)=(ax)*8</p>
<p> add ax,bx ;(ax)=(ax)*2+(ax)*8</p>
<p> mov ax,4c00h</p>
<p> int 21h</p>
<p>code ends</p>
<p>end start</p>
<p> </p>
<p>;应用举例:计算ffh*10</p>
<p>assume cs:code</p>
<p>code segment</p>
<p>start: mov ax,0ffh</p>
<p> mov bx,ax</p>
<p> shl ax,1 ;左移1位(ax)=(ax)*2</p>
<p> mov cl,3</p>
<p> shl bx,cl ;左移3位(bx)=(ax)*8</p>
<p> add ax,bx ;(ax)=(ax)*2+(ax)*8</p>
<p> mov ax,4c00h</p>
<p> int 21h</p>
<p>code ends</p>
<p>end start</p>
<p> </p>
<p>PS:</p>
<p>左移1位,N=(N)*2</p>
<p>左移2位,N=(N)*4</p>
<p>左移3位,N=(N)*8</p>
<p>左移4位,N=(N)*16</p>
<p>左移5位,N=(N)*32</p>
<p> </p>
<h3>十五章 外中断</h3>
<p>检测点15.1</p>
<p>(1) 仔细分析一下书中的in9中断例程,看看是否可以精简一下?</p>
<p>其实在我们的int 9中断例程中,模拟int指令调用原int 9中断例程的程序段是可以精简的,因为在进入中断例程后,IF和TF都已置0,没有必要再进行设置了,对于程序段:</p>
<p> pushf ;标志寄存器入栈</p>
<p> pushf</p>
<p> pop bx</p>
<p> and bh,11111100b ;IF和TF为flag的第9位和第8位</p>
<p> push bx</p>
<p> popf ;TF=0,IF=0</p>
<p> call dword ptr ds: ;CS、IP入栈;(IP)=ds:,(CS)=ds:</p>
<p> </p>
<p>可以精简为:</p>
<p> <span style="text-decoration: underline">pushf</span><span style="text-decoration: underline"> </span> ;标志寄存器入栈</p>
<p> <span style="text-decoration: underline">call dword ptr ds:</span> ;CS、IP入栈;(IP)=ds:,(CS)=ds:</p>
<p>两条指令。</p>
<p> </p>
<p>检测点15.1</p>
<p>(2) 仔细分析程序中的主程序,看看有什么潜在的问题?</p>
<p>在主程序中,如果在设置执行设置int 9中断例程的段地址和偏移地址的指令之间发生了键盘中段,则CPU将转去一个错误的地址执行,将发生错误。</p>
<p>找出这样的程序段,改写他们,排除潜在的问题。</p>
<p> </p>
<p>;在中断向量表中设置新的int 9中断例程的入口地址</p>
<p> cli ;设置IF=0屏蔽中断</p>
<p> mov word ptr es:,offset int9</p>
<p> mov es:,cs</p>
<p> sti ;设置IF=1不屏蔽中断</p>
<p> </p>
<p> </p>
<p>============更改后的int 9中断例程================</p>
<p>;功能:在屏幕中间依次显示'a'~'z',并让人看清。在显示过程中按下Esc键后,改变显示的颜色。</p>
<p>assume cs:code</p>
<p>stack segment</p>
<p> db 128 dup (0)</p>
<p>stack ends</p>
<p>data segment</p>
<p> dw 0,0</p>
<p>data ends</p>
<p>code segment</p>
<p>start: mov ax,stack</p>
<p> mov ss,ax</p>
<p> mov sp,128</p>
<p> </p>
<p>;将原来的int 9中断例程的入口地址保存在ds:0、ds:2单元中</p>
<p> mov ax,data</p>
<p> mov ds,ax</p>
<p> </p>
<p> mov ax,0</p>
<p> mov es,ax</p>
<p> </p>
<p> push es:</p>
<p> pop ds:</p>
<p> push es:</p>
<p> pop ds:</p>
<p> </p>
<p>;在中断向量表中设置新的int 9中断例程的入口地址</p>
<p> cli ;设置IF=0屏蔽中断</p>
<p> mov word ptr es:,offset int9</p>
<p> mov word ptr es:,cs</p>
<p> sti ;设置IF=1不屏蔽中断</p>
<p> </p>
<p>;依次显示'a'~'z'</p>
<p> mov ax,0b800h</p>
<p> mov es,ax</p>
<p> mov ah,'a'</p>
<p>s: mov es:,ah ;第12行第40列</p>
<p> inc ah</p>
<p> cmp ah,'z'</p>
<p> jnb s</p>
<p> </p>
<p>;将中断向量表中int 9中断例程的入口恢复为原来的地址</p>
<p> mov ax,0</p>
<p> mov es,ax</p>
<p> </p>
<p> push ds:</p>
<p> pop ss:</p>
<p> push ds:</p>
<p> pop es:</p>
<p> </p>
<p>;结束</p>
<p> mov ax,4c00h</p>
<p> int 21h</p>
<p> </p>
<p>;循环延时,循环100000h次</p>
<p>delay: push ax</p>
<p> push dx</p>
<p> mov dx,1000h</p>
<p> mov ax,0</p>
<p>delay1: sub ax,1</p>
<p> sbb dx,0 ;(dx)=(dx)-0-CF</p>
<p> cmp ax,0</p>
<p> jne delay1</p>
<p> cmp dx,0</p>
<p> jne delay1</p>
<p> pop dx</p>
<p> pop ax</p>
<p> ret</p>
<p> </p>
<p>;以下为新的int 9中断例程</p>
<p>int9: push ax</p>
<p> push bx</p>
<p> push es</p>
<p> </p>
<p> in al,60h ;从端口60h读出键盘输入</p>
<p> </p>
<p>;对int指令进行模拟,调用原来的int 9中断例程</p>
<p> pushf ;标志寄存器入栈</p>
<p> call dword ptr ds: ;CS、IP入栈;(IP)=ds:,(CS)=ds:</p>
<p> </p>
<p>;如果是ESC扫描码,改变显示颜色</p>
<p> cmp al,1 ;和esc的扫描码01比较</p>
<p> jne int9ret ;不等于esc时转移</p>
<p> </p>
<p> mov ax,0b800h</p>
<p> mov es,ax</p>
<p> inc byte ptr es: ;将属性值+1,改变颜色</p>
<p> </p>
<p>int9ret:pop es</p>
<p> pop bx</p>
<p> pop ax</p>
<p> iret</p>
<p> </p>
<p>code ends</p>
<p>end start</p>
<p> </p>
<h3>第十六章 直接定址表</h3>
<p>检测点16.1(两个程序)</p>
<p>下面的程序将code段中a处的8个数值累加,结果存储到b处的双字节中,补全程序。</p>
<p>程序一:</p>
<p>assume cs:code</p>
<p>code segment</p>
<p> a dw 1,2,3,4,5,6,7,8</p>
<p> b dd 0</p>
<p>start: mov si,0</p>
<p> mov cx,8</p>
<p>s: mov ax,<span style="text-decoration: underline">a</span><span style="text-decoration: underline"></span></p>
<p> add <span style="text-decoration: underline">a</span>,ax</p>
<p> adc <span style="text-decoration: underline">a</span>,0</p>
<p> add si,<span style="text-decoration: underline">2</span></p>
<p> loop s</p>
<p> mov ax,4c00h</p>
<p> int 21h</p>
<p>code ends</p>
<p>end start</p>
<p> </p>
<p>C:\DOCUME~1\SNUSER>debug 16-1.exe</p>
<p>-u</p>
<p>0C4E:0014 BE0000 MOV SI,0000</p>
<p>0C4E:0017 B90800 MOV CX,0008</p>
<p>0C4E:001A 2E CS:</p>
<p>0C4E:001B 8B840000 MOV AX,</p>
<p>0C4E:001F 2E CS:</p>
<p>0C4E:0020 01061000 ADD ,AX</p>
<p>0C4E:0024 2E CS:</p>
<p>0C4E:0025 8316120000 ADC WORD PTR ,+00</p>
<p>0C4E:002A 83C602 ADD SI,+02</p>
<p>0C4E:002D E2EB LOOP 001A</p>
<p>0C4E:002F B8004C MOV AX,4C00</p>
<p>0C4E:0032 CD21 INT 21</p>
<p>-r</p>
<p>AX=0000 BX=0000 CX=0034 DX=0000 SP=0000 BP=0000 SI=0000 DI=0000</p>
<p>DS=0C3E ES=0C3E SS=0C4E CS=0C4E IP=0014 NV UP EI PL NZ NA PO NC</p>
<p>0C4E:0014 BE0000 MOV SI,0000</p>
<p>-d 0c4e:0 1f</p>
<p>0C4E:0000 01 00 02 00 03 00 04 00-05 00 06 00 07 00 08 00 ................</p>
<p>0C4E:0010 00 00 00 00 BE 00 00 B9-08 00 2E 8B 84 00 00 2E ................</p>
<p>-g002f</p>
<p>AX=0008 BX=0000 CX=0000 DX=0000 SP=0000 BP=0000 SI=0010 DI=0000</p>
<p>DS=0C3E ES=0C3E SS=0C4E CS=0C4E IP=002F NV UP EI PL NZ AC PO NC</p>
<p>0C4E:002F B8004C MOV AX,4C00</p>
<p>-d 0c4e:0 1f</p>
<p>0C4E:0000 01 00 02 00 03 00 04 00-05 00 06 00 07 00 08 00 ................</p>
<p>0C4E:0010 24 00 00 00 BE 00 00 B9-08 00 2E 8B 84 00 00 2E $...............</p>
<p>-</p>
<p> </p>
<p>程序二:</p>
<p>assume cs:code</p>
<p>code segment</p>
<p> a dw 1,2,3,4,5,6,7,8</p>
<p> b dd 0</p>
<p>start: mov si,0</p>
<p> mov cx,8</p>
<p>s: mov ax,<span style="text-decoration: underline">a</span><span style="text-decoration: underline"></span></p>
<p> add <span style="text-decoration: underline">word ptr b</span>,ax</p>
<p> adc <span style="text-decoration: underline">word ptr b</span>,0</p>
<p> add si,<span style="text-decoration: underline">2</span></p>
<p> loop s</p>
<p> mov ax,4c00h</p>
<p> int 21h</p>
<p>code ends</p>
<p>end start</p>
<p> </p>
<p>检测点16.2</p>
<p> </p>
<p>下面的程序将data段中a处的8个数值累加,结果存储到b处的双字节中,补全程序。</p>
<p> </p>
<p>assume cs:code,es:data</p>
<p> </p>
<p>data segment</p>
<p> a db 1,2,3,4,5,6,7,8</p>
<p> b dw 0</p>
<p>data ends</p>
<p> </p>
<p>code segment</p>
<p>start:<span style="text-decoration: underline"> mov ax,data </span></p>
<p> <span style="text-decoration: underline">mov</span><span style="text-decoration: underline"> es,ax </span></p>
<p> mov si,0</p>
<p> mov cx,8</p>
<p>s: mov al,a</p>
<p> mov ah,0</p>
<p> add b,ax</p>
<p> inc si</p>
<p> loop s</p>
<p> </p>
<p> mov ax,4c00h</p>
<p> int 21h</p>
<p>code ends</p>
<p>end start</p>
<p> </p>
<p> </p>
<p>C:\DOCUME~1\SNUSER>debug 16-2.exe</p>
<p>-u</p>
<p>0C4F:0000 B84E0C MOV AX,0C4E</p>
<p>0C4F:0003 8EC0 MOV ES,AX</p>
<p>0C4F:0005 BE0000 MOV SI,0000</p>
<p>0C4F:0008 B90800 MOV CX,0008</p>
<p>0C4F:000B 26 ES:</p>
<p>0C4F:000C 8A840000 MOV AL,</p>
<p>0C4F:0010 B400 MOV AH,00</p>
<p>0C4F:0012 26 ES:</p>
<p>0C4F:0013 01060800 ADD ,AX</p>
<p>0C4F:0017 46 INC SI</p>
<p>0C4F:0018 E2F1 LOOP 000B</p>
<p>0C4F:001A B8004C MOV AX,4C00</p>
<p>0C4F:001D CD21 INT 21</p>
<p>0C4F:001F 56 PUSH SI</p>
<p>-r</p>
<p>AX=0000 BX=0000 CX=002F DX=0000 SP=0000 BP=0000 SI=0000 DI=0000</p>
<p>DS=0C3E ES=0C3E SS=0C4E CS=0C4F IP=0000 NV UP EI PL NZ NA PO NC</p>
<p>0C4F:0000 B84E0C MOV AX,0C4E</p>
<p>-d 0c4e:0 f</p>
<p>0C4E:0000 01 02 03 04 05 06 07 08-00 00 00 00 00 00 00 00 ................</p>
<p>-g001a</p>
<p>AX=0008 BX=0000 CX=0000 DX=0000 SP=0000 BP=0000 SI=0008 DI=0000</p>
<p>DS=0C3E ES=0C4E SS=0C4E CS=0C4F IP=001A NV UP EI PL NZ NA PO NC</p>
<p>0C4F:001A B8004C MOV AX,4C00</p>
<p>-d 0c4e:0 f</p>
<p>0C4E:0000 01 02 03 04 05 06 07 08-24 00 00 00 00 00 00 00 ........$.......</p>
<p>-</p>
<p> </p>
<h3>第十七章 使用BIOS进行键盘输入和磁盘读写</h3>
<p> </p>
<p>检测点17.1</p>
<p> </p>
<p>“在int 16h中断例程中,一定有设置IF=1的指令。”这种说法对吗?</p>
<p> </p>
<p>正确,当键盘缓冲区为空时,如果设置IF=0,int 9中断无法执行,循环等待会死锁。</p>
<p> </p>
<p> </p>
<p> </p>
<p>相关内容:</p>
<p>IF=1,CPU响应中断,引发中断过程 </p>
<p>IF=0,不响应可屏蔽中断 </p>
<p>几乎所有由外设引发的外中断,都是可屏蔽中断(int 9是可屏蔽中断)</p>
<p> </p>
<p>CPU对外设输入的通常处理方法:</p>
<p>(1)外设的输入端口</p>
<p>(2)向CPU发出外中断(可屏蔽中断)信息</p>
<p>(3)CPU检测到可屏弊中断信息,如果IF=1,cpu在执行完当前指令后响应中断,执行相应的中断例程</p>
<p>(4)可在中断例程中实现对外设输入的处理</p>
<p> </p>
<p>由于本人水平有限,制作仓促,不能保证解析完全正确。</p>
<p>如果你在对照的过程中,发现了错误的地方,可以留言告知本人,在此先表示感谢。</p>
<p>制作不易 请点赞支持!</p>
</div>
</div><br><br>
来源:https://www.cnblogs.com/liuhanxu/p/13784010.html
頁:
[1]